C#多執行緒中訪問winform控制元件
阿新 • • 發佈:2019-01-27
方法一:System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;
不推薦使用這種方式,禁止編譯器對跨執行緒訪問做檢查的方式實現。
方法二:使用delegate和invoke
private delegate void UpdateLabel2Delegate(string message); void UpdateLabel2(string message) { if (label1.InvokeRequired) { UpdateLabel2Delegate md = new UpdateLabel2Delegate(UpdateLabel2); label1.Invoke(md, new object[] { message }); //label1.BeginInvoke(md, new object[] { message }); } else { label1.Text = message; } }
或者:
void UpdateLabel2(string message)
{
this.Invoke((EventHandler)delegate{ this.label1.Text = message;});
}
或者:(不使用委託方式)
方法三:使用BackgroundWorker元件private void button2_Click(object sender, EventArgs e) { Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel2)); thread1.Start("更新Label"); } private void UpdateLabel2(object str) { if (label2.InvokeRequired) { // 當一個控制元件的InvokeRequired屬性值為真時,說明有一個建立它以外的執行緒想訪問它 Action<string> actionDelegate = (x) => { this.label2.Text = x.ToString(); }; // 或者 // Action<string> actionDelegate = delegate(string txt) { this.label2.Text = txt; }; this.label2.Invoke(actionDelegate, str); } else { this.label2.Text = str.ToString(); } }
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(); }