使用中:
c#写界面程序中经常会出现子线程访问主线程控件的情况,在不注意的情况下就会弹出错误提示。。。(截图就不截图了,太常见了)
比如你这样类型的写着:
public Form1()
{
InitializeComponent();
}
private void btnStart_Click(object sender, EventArgs e)
{
Thread newThread = new Thread(new ThreadStart(Count));
newThread .IsBackground = true;//后台线程,前台线程GG,它也GG
newThread .Start();
}
public void Count()
{
for (int i = 0; i < 100; i++)
{
lblCount.Text = i.ToString();//此时就会报出“线程间操作无效: 从不是创建控件" lblCount" 的线程访问它”;
Thread.Sleep(1000);
}
}
处理:
网上有很多处理方式,就不一一列举了,说一下最常用的那就是Invoke或BeginInvoke了.
public Form1()
{
InitializeComponent();
}
private void btnStart_Click(object sender, EventArgs e)
{
Thread newThread = new Thread(new ThreadStart(Count));
newThread .IsBackground = true;//后台线程,前台线程GG,它也GG
newThread .Start();
}
public void Count()
{
for (int i = 0; i < 100; i++)
{
this.Invoke((EventHandler)(delegate
{
lblCount.Text = i.ToString();//此时就会报出“线程间操作无效: 从不是创建控件" lblCount" 的线程访问它”;
Thread.Sleep(1000);
}
}
}
这样就可以解决跨线程访问控件问题了,Invoke是同步的方式,BeginInvoke是异步的方式,看情况使用。
还有一种比较简洁的写法,如下:
public void Count()
{
for (int i = 0; i < 100; i++)
{
Thread.Sleep(1000);
//异步委托方式更新UI,同步委托需将BeginInvoke变Invoke
this.BeginInvoke(new Action(() =>
{
lblCount.Text = i.ToString();
}));
}
}
如果想要带参数的话可以:
public void Count()
{
for (int i = 0; i < 100; i++)
{
Thread.Sleep(1000);
this.BeginInvoke(mydelegate, new object[] { i });
}
}
public void ShowMessage(int i)
{
lblCount.Text = i.ToString();
}
正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会