最近在做一个winform的小软件(抢票的...)。登录窗体要从远程web页面获取一些数据,为了不阻塞登录窗体的显示,开了一个线程去加载数据远程的数据,会报一个错误"线程间操作无效: 从不是创建控件“lbl_loading_msg”的线程访问它。"百度一下,原来从.net framework 2.0开始,为了安全,不允许跨线程操作控件。解决办法如下:
1、声明一个拥有委托类型的方法,作为代理 操作控件的代码:
/// <summary>
/// 在线程中操作窗体的控件
/// </summary>
/// <param name="action"></param>
private void OpeMainFormControl(Action action)
{
if (this.InvokeRequired)
{
this.Invoke(action); //返回主线程(创建控件的线程)
}
else
{
action();
}
}
2、然后在线程中,只要是对控件的操作都委托给上面的方法执行,不管是读取还是赋值:
Thread t = new Thread(new ThreadStart(delegate()
{
//执行一段费时的代码
//......
//线程中操作 控件,委托给invoker
OpeMainFormControl(delegate()
{
this.lbl_loading_msg.Visible = false; //隐藏验证码加载中...的提示
this.toolStripStatusLabel1.Text = ""; //清空状态栏
this.btnlogin.Enabled = true; //启用登录按钮
});
}));
t.Start(); //启动线程
上面代码需要注意的是:仅对控件操作的代码委托给this.invoker。上面委托操作的方法中的this是窗体对象,也就是创建控件的线程,this.invoker(action)就是将代码交给它来执行。这样就操作控件的代码就交给了创建控件的线程来执行了。我们不要将费时操作的代码也交给this.invoker来执行,不然就失去了线程的意义了(相当于没有开线程)。this.inoker又回到了主线程,会产生阻塞!下面代码会阻塞:
//将费时的代码也委托给 主线程(创建控件的线程)来执行,会阻塞窗体
OpeMainFormControl(delegate()
{
//执行一段费时的代码
//......
//线程中操作 控件,委托给invoker
this.lbl_loading_msg.Visible = false; //隐藏验证码加载中...的提示
this.toolStripStatusLabel1.Text = ""; //清空状态栏
this.btnlogin.Enabled = true; //启用登录按钮
});
还有一点:如果对控件的操作代码没有委托给创建它的线程来执行(比如实例化另外一个窗口,show出来),它的一些设置会不起作用,比如:窗体初始显示位置等...