线程间操作无效: 从不是创建控件“button1”的线程访问它。
2015-04-17 23:05 糯米粥 阅读(19530) 评论(1) 编辑 收藏 举报.net2后是不能跨线程访问控件的。,窗体上的控件是当前线程创建的,当用户异步执行一个方法:在该方法中给窗体上的控件赋值,记住:当执行一个异步委托的时候,其实
就是开了一个线程去执行那个方法,这样就会报错:线程间操作无效: 从不是创建控件“某某某”的线程访问它。
C# WinForm开 发中,这是一个比较常见的异常:线程间操作无效,从不是创建控件“xxx”的线程访问它。这个异常来源于.NET2的一个限制:工作线程不能访问窗口线程 创建的控件。解决方法主要有两种,一种是在窗口线程中设置CheckForIllegalCrossThreadCalls = false ;另一种方式比较麻烦,使用委托的方式调用Invoke方法。
public Form1() { InitializeComponent(); Control.CheckForIllegalCrossThreadCalls = false; }
但以上不是推荐的方法。更好的办法是用委托解决
private void button1_Click(object sender, EventArgs e) { new Action(show).BeginInvoke(null, null); } void show() { //异步外的方法。这样窗体不会假死 while (true) { Thread.Sleep(2000); Action ac = new Action(showText); this.Invoke(ac); //在同步方法里面实现更新窗体上的数据 } } /// <summary> /// 更新数据 /// </summary> void showText() { richTextBox1.AppendText("更新\n"); }
或者使用InvokeRequired属性判断
/*
// 摘要:
// 获取一个值,该值指示调用方在对控件进行方法调用时是否必须调用 Invoke 方法,因为调用方位于创建控件所在的线程以外的线程中。
//
// 返回结果:
// 如果控件的 System.Windows.Forms.Control.Handle 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke
// 方法对控件进行调用),则为 true;否则为 false。
private void button1_Click(object sender, EventArgs e) { //new Action(show).BeginInvoke(null, null); new Action(show1).BeginInvoke(null, null); } void show1() { while (true) { Thread.Sleep(2000);//模拟等待效果 show2(); } } void show2() { //说明的当前外部线程 /* // 摘要: // 获取一个值,该值指示调用方在对控件进行方法调用时是否必须调用 Invoke 方法,因为调用方位于创建控件所在的线程以外的线程中。 // // 返回结果: // 如果控件的 System.Windows.Forms.Control.Handle 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke // 方法对控件进行调用),则为 true;否则为 false。 */ if (InvokeRequired) { /*既然是外部线程,那么就没有权限访问主线程上的控件 * 故要主线程访问,开启一个异步委托捆绑要执行的方法 * 交给主线程执行 */ Action ac = new Action(show2); this.Invoke(ac); //这里执行后。则InvokeRequired就为false。因为此时已经是主线程访问当前创建的控件 } else { richTextBox1.AppendText("更新77\n"); } }
看了第一段代码是不是很不爽的感觉。showText()方法就一条赋值语句,则也独立成一个方法。这里可以简化代码,用匿名函数或者用更简单的lambda表达式(这个方法不需要其他用户调用,就可以考虑用匿名函数)。那么看简化后的代码
private void button1_Click(object sender, EventArgs e) { new Action(show).BeginInvoke(null, null); } void show() { //异步外的方法。这样窗体不会假死 while (true) { Thread.Sleep(2000);//在异步方法外实现等待,这样窗体不会假死 //Action ac = new Action(showText); //this.Invoke(ac); //在同步方法里面实现更新窗体上的数据 //匿名函数 Action at = new Action(delegate(){ richTextBox1.AppendText("更新\n"); }); //lambda表达式更简单 Action at1 = new Action(()=> { richTextBox1.AppendText("更新\n"); }); this.Invoke(at);
//这里this。代表当前窗体(控件)上的线程执行此方法
//这里不一定是this,只要是当前窗体上的控件都可以,比如
// button1.Invoke(at);
this.Invoke(at);//Invoke:在拥有此控件的基础窗口句柄的线程上执行指定的委托。
//richTextBox1.Invoke(at);
//button1.Invoke(at);
} }
我们用同样的方法把InvokeRequired属性判断的那段代码也用lambda表达式简写
private void button1_Click(object sender, EventArgs e) { new Action(show1).BeginInvoke(null, null); } void show1() { //while (true) //{ // Thread.Sleep(2000);//模拟等待效果 // show2(); //} for (int i = 0; i < 5000; i++) { Thread.Sleep(2000);//模拟等待效果 //show2(); if (InvokeRequired) { Action ac = new Action(() => { richTextBox1.AppendText("更新767\n"); }); this.Invoke(ac); //这里执行后。则InvokeRequired就为false。因为此时已经是主线程访问当前创建的控件 } } }
以上两种方法都是实现相同的效果,差别就是多了一个InvokeRequired判断是否需要执行Invoke()方法
参考:http://www.cnblogs.com/txw1958/archive/2012/08/21/2649192.html