线程访问UI控件和Control.CheckForIllegalCrossThreadCalls属性

发布时间:2020-06-20 20:12:02 作者:cnn237111
来源:网络 阅读:4825

C#的Winform开发中,一般是不推荐使用线程去直接访问UI控件的。

访问 Windows 窗体控件本质上不是线程安全的。如果有两个或多个线程操作某一控件的状态,则可能会迫使该控件进入一种不一致的状态。还可能出现其他与线程相关的 bug,包括争用情况和死锁。但是有时在程序编写的时候写了一些线程直接访问UI控件的代码,运行时也不会报错,就以为这样做是可以的。比如下面的代码,定义一个线程,并且在运行是访问进度条控件。运行代码后发现,一切都ok,没有任何异常抛出。

  1. private void button1_Click(object sender, EventArgs e) 
  2.         { 
  3.             Thread t = new Thread(() => 
  4.             { 
  5.                 try 
  6.                 { 
  7.                     for (int i = 1; i <= 100; i++) 
  8.                     { 
  9.                         progressBar1.Value = i; 
  10.                         Thread.Sleep(100); 
  11.                     } 
  12.                 } 
  13.                 catch (Exception ex) 
  14.                 { 
  15.                     MessageBox.Show(ex.Message); 
  16.                 } 
  17.             }); 
  18.             t.Start(); 

 

但是在调试的时候,却会捕获一个异常InvalidOperationException,并提示消息:“线程间操作无效,从不是创建控件的线程访问它。”这点困惑了很多人。其实这一切都是和Control.CheckForIllegalCrossThreadCalls有关。注意: Control.CheckForIllegalCrossThreadCalls属性是在 .NET Framework 2.0 版中新增的。它的作用是获取或设置一个值,该值指示是否捕获对错误线程的调用。如果设为true则会捕获对错误线程的调用,反之亦反。

如果一个线程并没有创建控件A而去访问控件A,有时候会很幸运没什么错误,但是在一些复杂的情况下会导致不可以预料的错误。因此将 CheckForIllegalCrossThreadCalls 设置为 true 可以更容易地查找并诊断此线程活动。

CheckForIllegalCrossThreadCalls会在Control类的静态构造方法中调用如下的语句,Debugger.IsAttached的值表示调试器是否附到进程中:

  1. static Control() 
  2.     //... 
  3.     checkForIllegalCrossThreadCalls = Debugger.IsAttached; 
  4.     //... 

因此,当debug的时候,控件的checkForIllegalCrossThreadCalls是true。但是运行的时候checkForIllegalCrossThreadCalls是false。

我们可以手动的设置该值,以此控制是否需要捕获异常。如下的代码:   

  1. public partial class Form1 : Form 
  2.    { 
  3.        public Form1() 
  4.        { 
  5.            InitializeComponent(); 
  6.        } 
  7.  
  8.        private void Form1_Load(object sender, EventArgs e) 
  9.        { 
  10.            showCurrentCheckForIllegalCrossThreadCallsValue(); 
  11.        } 
  12.  
  13.        private void button1_Click(object sender, EventArgs e) 
  14.        { 
  15.            Thread t = new Thread(() => 
  16.            { 
  17.                try 
  18.                { 
  19.                    for (int i = 1; i <= 100; i++) 
  20.                    { 
  21.                        progressBar1.Value = i; 
  22.                        Thread.Sleep(100); 
  23.                    } 
  24.                } 
  25.                catch (Exception ex) 
  26.                { 
  27.                    MessageBox.Show(ex.Message); 
  28.                } 
  29.            }); 
  30.            t.Start(); 
  31.        } 
  32.  
  33.        private void showCurrentCheckForIllegalCrossThreadCallsValue() 
  34.        { 
  35.            label1.Text = "ProgressBar's CheckForIllegalCrossThreadCalls Value:" + ProgressBar.CheckForIllegalCrossThreadCalls.ToString(); 
  36.        } 
  37.  
  38.        private void button2_Click(object sender, EventArgs e) 
  39.        { 
  40.            ProgressBar.CheckForIllegalCrossThreadCalls = true
  41.            showCurrentCheckForIllegalCrossThreadCallsValue(); 
  42.        } 
  43.  
  44.        private void button3_Click(object sender, EventArgs e) 
  45.        { 
  46.            ProgressBar.CheckForIllegalCrossThreadCalls = false
  47.            showCurrentCheckForIllegalCrossThreadCallsValue(); 
  48.        } 
  49.    } 

 

程序运行的时候(非调试),ProcessBar当前的CheckForIllegalCrossThreadCalls为False,手动设置为True后,再调用线程访问UI,得到了异常。

多做几次实验就明白了。

线程访问UI控件和Control.CheckForIllegalCrossThreadCalls属性

最后想说明的是,要在线程中访问UI控件,最好使用BackgroundWorker类。

推荐阅读:
  1. 如何使用JavaScript UI控件(WijmoJS)构建
  2. iOS UI基本控件UIButton介绍

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

多线程 ui

上一篇:java sql编辑器 动态报表 数据库备份还原quartz

下一篇:php魔术方法

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》