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

在利用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.        }  
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();

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.        }  
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......"; } }

 

转载于:https://www.cnblogs.com/freedom831215/archive/2010/08/09/1795960.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值