在WinForms程序开发中,有时使用多线程或者异步处理比较耗时的数据,处理完成后需要把结果反馈到程序界面上,这是就需要操作WinForms程序的界面控件了。
如果直接操作的话,则会出现以下错误
注:这个异常来源于.NET2的一个限制:工作线程不能访问窗口线程创建的控件。
线程间操作无效: 从不是创建控件“updateMaterial”的线程访问它。
解决方法有两种:
方法一:是在窗口线程中设置CheckForIllegalCrossThreadCalls = false (不推荐)
比如窗口中有一个button1,我要新建一个线程访问到button1。
this.button1.Enabled = false;
new Thread(new ThreadStart(delegate()
{
try {
CheckForIllegalCrossThreadCalls = false;
// 直接设置会引发异常:线程间操作无效,从不创建控件的线程访问它
this.button1.Enabled = true;
}catch(Exception ex)
{
MessageBox.Show(ex.ToString());
}
})).
方法二:使用委托的方式Invoke操作界面的控件( 推荐)
private delegate void _SetButtonState(Button button,bool state);
private void SetButtonState(Button button,bool state)
{
// 对于该控件的请求来自于创建该控件所在线程以外的线程
if (button.InvokeRequired)
{
_SetButtonState _set = new _SetButtonState(delegate(Button _button, bool _state)
{
_button.Enabled = _state;
});
this.Invoke(_set, button, state);
}
else
{
button.Enabled = state;
}
}
创建一个线程调用这个委托
this.button1.Enabled = false;
new Thread(new ThreadStart(delegate()
{
try {
// 直接设置会引发异常:线程间操作无效,从不创建控件的线程访问它
// CheckForIllegalCrossThreadCalls = false;
// this.button1.Enabled = true;
// 用委托的方式
SetButtonState(button1,true);
}catch(Exception ex)
{
MessageBox.Show(ex.ToString());
}
})).Start();
另外,附上一个修改的更为直接的Invoke函数
private delegate void _SetListData();
private void SetListData()
{
_SetListData _set = new _SetListData(delegate()
{
this.t_material_infoTableAdapter.Fill(this.materialDataSet.t_material_info);
});
this.Invoke(_set);
}
SetListData();