问题:运行一下代码:报错 大概意思就是跨线程操作,label没法儿赋值
1.private void mainloop(){
2.while (threadrun)
3. {4.
5 . Thread.Sleep(50);
6 . Label.Text = MainForm._comPort.JY.cs.satcount.ToString();
7 . }
8.}
解决方案:
一、在5-6行中间添加以下代码:
CheckForIllegalCrossThreadCalls = false;//禁止编译器对跨线程访问做检查
二、使用deletgate invoke来从其他线程调用
if (Label.InvokeRequired) { // 当一个控件的InvokeRequired属性值为真时,说明有一个创建它以外的线程想访问它 Action<string> actionDelegate = (x) => { this.Label.Text = x.ToString(); }; // 或者 // Action<string> actionDelegate = delegate(string txt) { this.label2.Text = txt; }; this.Label.Invoke(actionDelegate, str); } else { this.Label.Text = str.ToString(); }
或者如下invoke:
this.Invoke(new Action(()=>
{
Label.Text = MainForm._comPort.JY.cs.satcount.ToString();
}));
或者单独使用delegate,详细参见delegate使用
三、
使用deletgate BeginInvoke来从其他线程调用
Invoke方法和BeginInvoke方法的区别是
Invoke方法是同步的, 它会等待工作线程完成,
BeginInvoke方法是异步的, 它会另起一个线程去完成工作线程
四、BackgroundWorker 组件:(此方式本人暂未尝试使用)
BackgroundWorker是.NET里面用来执行多线程任务的控件,它允许编程者在一个单独的线程上执行一些操作。耗时的操作(如下载和数据库事务)。用法简单
private void button4_Click(object sender, EventArgs e) { using (BackgroundWorker bw = new BackgroundWorker()) { bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted); bw.DoWork += new DoWorkEventHandler(bw_DoWork); bw.RunWorkerAsync("Tank"); } } void bw_DoWork(object sender, DoWorkEventArgs e) { // 这里是后台线程, 是在另一个线程上完成的 // 这里是真正做事的工作线程 // 可以在这里做一些费时的,复杂的操作 Thread.Sleep(5000); e.Result = e.Argument + "工作线程完成"; } void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { //这时后台线程已经完成,并返回了主线程,所以可以直接使用UI控件了 this.label4.Text = e.Result.ToString(); }