WinForms程序使用委托(Invoke)的方式操作界面的控件

来源:互联网 发布:python高斯分布散点图 编辑:程序博客网 时间:2024/04/30 22:56

在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();


原创粉丝点击