c#不允许对跨线程的控件的访问,如需操作跨线程的操作需要通过委托(delegate),即是函数指针来操作跨线程。
在多线程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法时会报错的,Invoke 和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。 正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。 而所谓的一面响应操作,一面添加节点永远只能是相对的,使 UI 线程的负担不至于太大而已,因为界面的正确更新始终要通过 UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到 UI 线程中去做,这样也就达到了减轻 UI 线程负担的目的了。 using System.Threading; //启动一个线程 Thread thread=new Thread(new ThreadStart(DoWork)); 正确的做法是用Invoke\BeginInvokeusing System.Threading; namespace MsgConsole { public partial class Form1 : Form { //定义委托 public delegate void SetGridViewCallback(); public Form1() { Form.CheckForIllegalCrossThreadCalls = false; InitializeComponent(); } private void bt_check_Click(object sender, EventArgs e) { Thread Msgthread = new Thread(new ThreadStart(GetMsgLog)); Msgthread.IsBackground = true; Msgthread.Start(); } public void GetMsgLog() { while (IsStop == false) { try { if (dataGridView1.InvokeRequired) //控件是否跨线程?如果是,则执行括号里代码 { SetGridViewCallback setGridCallback = new SetGridViewCallback(BindData); //实例化委托对象 this.BeginInvoke(setGridCallback); } Thread.Sleep(_timespan); } catch (Exception ex) { MessageBox.Show(ex.Message); } finally { } } } public void BindData() { dataGridView1.Columns.Clear(); this.dataGridView1.DataSource = cmsg.list; } }