在利用Visual Studio2005编写窗体控件的应用程序时,往往会遇到这样的问题:有两个控件,分别为A和B,我们要通过控件A做一个复杂操作,在这个操作过程要设置B的相关属性。如果我们把该操作放在线程中,就可能出现线程安全问题。下面根据一个例子来说明一下。
我们假设一个情景,窗体中有一个TabControl控件tc_thread,控件里边共有两个tab页分别为tab1和tab2,控件的下边有一个label控件myLabel。我们希望当来回切换tab1和tab2时,myLabel的text能做出响应,整个过程要放在一个线程中。代码如下
- private void tc_thread_SelectedIndexChanged(object sender, EventArgs e)
- {
- if (tc_timer.SelectedTab.Equals(tab1))
- tab1.text = "showing tab1......";
- else if (tc_timer.SelectedTab.Equals(tab2))
- {
- try
- {
- Thread myThread = new Thread(new ThreadStart(SetLabel));
- myThread.Start();
- }
- catch (Exception ex)
- {
- MessageBox.Show(ex.Message);
- }
- }
- }
- private void SetLabel()
- {
- myLabel.text = "showing tab2......";
- }
这时候如果运行程序,会出现异常:线程间操作无效: 从不是创建控件tc_thread的线程访问它。因为windows窗体控件不是线程安全的,如果几个线程操作某一控件的状态,可能会使该控件的状态不一致,出现争用或死锁状态。这种情况有以下解决办法:
1. 可以在load时将CheckForIllegalCrossThreadCalls 属性的值设置为 false 。这样进行非安全线程访问时,运行环境就不去检验它是否是线程安全的。
Control.CheckForIllegalCrossThreadCalls=false;
2. 利用委托机制实现线程安全。上面的代码可以更改如下:
建立一个委托:delegate void SetLabelCallBack();
- private void tc_thread_SelectedIndexChanged(object sender, EventArgs e)
- {
- if (tc_timer.SelectedTab.Equals(tab1))
- tab1.text = "showing tab1......";
- else if (tc_timer.SelectedTab.Equals(tab2))
- {
- try
- {
- Thread myThread = new Thread(new ThreadStart(SetLabel));
- myThread.Start();
- }
- catch (Exception ex)
- {
- MessageBox.Show(ex.Message);
- }
- }
- }
- private void SetLabel()
- {
- if(tc_thread.InvokeRequired)
- {
- SetLabelCallBack labDele = new SetLabelCallBack(SetLabel);
- this.Invoke(labDele, new object[]{});
- }
- else
- {
- myLabel.text = "showing tab2......";
- }
- }