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

在利用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 。这样进行非安全线程访问时,运行环境就不去检验它是否是线程安全的。

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值