C# 跨線程調用控件
線程間操作無效
界面上有一個button和一個label, 點擊button會啟動一個線程來更新Label的值
復制代碼
private void button1_Click(object sender, EventArgs e)
{
Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel));
thread1.Start("更新Label");
}
private void UpdateLabel(object str)
{
this.label1.Text = str.ToString();
}
復制代碼
運行後, 程序會報錯 "跨線程操作無效,從不是創建"label1"的線程訪問它"
這是因為.NET禁止了跨線程調用控件, 否則誰都可以操作控件,最後可能造成錯誤。
下面介紹幾種跨線程調用控件的方法
第一種辦法:禁止編譯器對跨線程訪問做檢查
這是最簡單的辦法, 相當於不檢查線程之間的沖突,允許各個線程隨便亂搞,最後Lable1控件的值是什麼就難以預料了 (不推薦使用這種方法)
public Form1()
{
InitializeComponent();
// 加入這行
Control.CheckForIllegalCrossThreadCalls = false;
}
第二種辦法: 使用delegate和invoke來從其他線程中調用控件
調用控件的invoke方法,就可以控制控件了,例如
復制代碼
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();
}
}
復制代碼
第三種辦法: 使用delegate和BeginInvoke來從其他線程中控制控件
只要把上面的 this.label2.Invoke(actionDelegate, str); 中的 Invoke 改為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();
}