线程间操作无效: 从不是创建控件“textBox1”的线程访问它
背景:通过一个辅助线程计算出的一个值赋给textBox1.text;
解决办法:
1.直接在窗体的构造函数中加:System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false; 此时把所有的控件
合法性线程检查全部都给禁止掉了。
2.通过代理来解决(msdn)
private delegate void SetTextCallback(string text);
//在给textBox1.text赋值的地方调用以下方法即可
private void SetText(string text)
{
// InvokeRequired需要比较调用线程ID和创建线程ID
// 如果它们不相同则返回true
if (this.textBox1.InvokeRequired)
{
SetTextCallback d = new SetTextCallback(SetText);
this.Invoke(d, new object[] { text });
}
else
{
this.textBox1.Text = text;
}
}
有两种方式解决
1.在窗体构造函数中写Control.CheckForIllegalCrossThreadCalls =false;
2.使用Invoke等委托函数。
问题原因是。net2.0以后拒绝多线程访问空间,避免空间造成死锁。以前Control.CheckForIllegalCrossThreadCalls =false;默认
就是这样,现在默认为true。
如果不会好几个线程同时操作一个控件用方法1就可以。如果存在多个线程一起操作控件使用方法2
方法2可以
Invoke(new MethodInvoker(delegate()
{
//do something... }));
可见,不同的调用方式,执行委托的线程也不一样。
如果直接调用委托,则委托代码在子线程中执行;
//实例化一个委托
MyDelegate dlgt = new MyDelegate(dlgtOutPut);
//
dlgtOutPut("直接调用委托");
如果使用Invoke来调用委托,则发现委托代码则在主线程中执行。确切地说,当使用Invoke调用委托时,则在拥有此控件的基
础窗口句柄的线程上执行委托。
为防止这种方法可以直接在调用委托方法时使用Invoke来调用