問題:運行一下代碼:報錯 大概意思就是跨線程操作,label沒法兒賦值
1.private void mainloop(){
2.while (threadrun)
3. {4.
5 . Thread.Sleep(50);
7 . }
8.}
解決方案:
一、在5-6行中間添加以下代碼:
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:
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(); }