线程间操作无效: 从不是创建控件“...”的线程访问它

在利用Visual Studio2005编写窗体控件的应用程序时,往往会遇到这样的问题:有两个控件,分别为A和B,我们要通过控件A做一个复杂操作,在这个操作过程要设置B的相关属性。如果我们把该操作放在线程中,就可能出现线程安全问题。下面根据一个例子来说明一下。

我们假设一个情景,窗体中有一个TabControl控件tc_thread,控件里边共有两个tab页分别为tab1和tab2,控件的下边有一个label控件myLabel。我们希望当来回切换tab1和tab2时,myLabel的text能做出响应,整个过程要放在一个线程中。代码如下

C-sharp代码
  1. private void tc_thread_SelectedIndexChanged(object sender, EventArgs e)   
  2.         {   
  3.             if (tc_timer.SelectedTab.Equals(tab1))   
  4.                 tab1.text = "showing tab1......";   
  5.             else if (tc_timer.SelectedTab.Equals(tab2))   
  6.             {   
  7.                 try  
  8.                 {   
  9.                     Thread myThread = new Thread(new ThreadStart(SetLabel));   
  10.                     myThread.Start();   
  11.                 }   
  12.                 catch (Exception ex)   
  13.                 {   
  14.                     MessageBox.Show(ex.Message);   
  15.                 }   
  16.             }   
  17.         }   
  18.   
  19. private void SetLabel()   
  20.        {   
  21.             myLabel.text = "showing tab2......";   
  22.        }  

 

这时候如果运行程序,会出现异常:线程间操作无效: 从不是创建控件tc_thread的线程访问它。因为windows窗体控件不是线程安全的,如果几个线程操作某一控件的状态,可能会使该控件的状态不一致,出现争用或死锁状态。这种情况有以下解决办法:

1. 可以在load时将CheckForIllegalCrossThreadCalls 属性的值设置为 false 。这样进行非安全线程访问时,运行环境就不去检验它是否是线程安全的。

Control.CheckForIllegalCrossThreadCalls=false;

2. 利用委托机制实现线程安全。上面的代码可以更改如下:

建立一个委托:delegate void SetLabelCallBack();

C-sharp代码
  1. private void tc_thread_SelectedIndexChanged(object sender, EventArgs e)   
  2.         {   
  3.             if (tc_timer.SelectedTab.Equals(tab1))   
  4.                 tab1.text = "showing tab1......";   
  5.             else if (tc_timer.SelectedTab.Equals(tab2))   
  6.             {   
  7.                 try  
  8.                 {   
  9.                     Thread myThread = new Thread(new ThreadStart(SetLabel));   
  10.                     myThread.Start();   
  11.                 }   
  12.                 catch (Exception ex)   
  13.                 {   
  14.                     MessageBox.Show(ex.Message);   
  15.                 }   
  16.             }   
  17.         }   
  18.   
  19. private void SetLabel()   
  20.        {   
  21.             if(tc_thread.InvokeRequired)   
  22.             {   
  23.                 SetLabelCallBack labDele = new SetLabelCallBack(SetLabel);   
  24.                 this.Invoke(labDele, new object[]{});   
  25.             }   
  26.             else  
  27.             {   
  28.                 myLabel.text = "showing tab2......";   
  29.             }   
  30.        }  

 

你可能感兴趣的:(线程)