C# 跨线程调用控件
在C# 的應(yīng)用程序開(kāi)發(fā)中, 我們經(jīng)常要把UI線程和工作線程分開(kāi),防止界面停止響應(yīng)。 ?同時(shí)我們又需要在工作線程中更新UI界面上的控件,
下面介紹幾種常用的方法
?
閱讀目錄
?
線程間操作無(wú)效
界面上有一個(gè)button和一個(gè)label, ?點(diǎn)擊button會(huì)啟動(dòng)一個(gè)線程來(lái)更新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();}運(yùn)行后, 程序會(huì)報(bào)錯(cuò) "跨線程操作無(wú)效,從不是創(chuàng)建"label1"的線程訪問(wèn)它"
這是因?yàn)?NET禁止了跨線程調(diào)用控件, 否則誰(shuí)都可以操作控件,最后可能造成錯(cuò)誤。 ??
?
下面介紹幾種跨線程調(diào)用控件的方法
?
第一種辦法:禁止編譯器對(duì)跨線程訪問(wèn)做檢查
這是最簡(jiǎn)單的辦法, 相當(dāng)于不檢查線程之間的沖突,允許各個(gè)線程隨便亂搞,最后Lable1控件的值是什么就難以預(yù)料了 (不推薦使用這種方法)
public Form1(){InitializeComponent();// 加入這行Control.CheckForIllegalCrossThreadCalls = false;}?
第二種辦法: 使用delegate和invoke來(lái)從其他線程中調(diào)用控件
調(diào)用控件的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){// 當(dāng)一個(gè)控件的InvokeRequired屬性值為真時(shí),說(shuō)明有一個(gè)創(chuàng)建它以外的線程想訪問(wèn)它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來(lái)從其他線程中控制控件
只要把上面的?this.label2.Invoke(actionDelegate, str); 中的 Invoke 改為BeginInvoke方法就可以了
Invoke方法和BeginInvoke方法的區(qū)別是
Invoke方法是同步的, 它會(huì)等待工作線程完成,
BeginInvoke方法是異步的, 它會(huì)另起一個(gè)線程去完成工作線程
?
第四種辦法: 使用BackgroundWorker組件(推薦使用這個(gè)方法)
BackgroundWorker是.NET里面用來(lái)執(zhí)行多線程任務(wù)的控件,它允許編程者在一個(gè)單獨(dú)的線程上執(zhí)行一些操作。耗時(shí)的操作(如下載和數(shù)據(jù)庫(kù)事務(wù))。用法簡(jiǎn)單?
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){ // 這里是后臺(tái)線程, 是在另一個(gè)線程上完成的// 這里是真正做事的工作線程// 可以在這里做一些費(fèi)時(shí)的,復(fù)雜的操作Thread.Sleep(5000);e.Result = e.Argument + "工作線程完成";}void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e){//這時(shí)后臺(tái)線程已經(jīng)完成,并返回了主線程,所以可以直接使用UI控件了 this.label4.Text = e.Result.ToString(); }?
總結(jié)
以上是生活随笔為你收集整理的C# 跨线程调用控件的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: MFC的GDI绘制坐标问题
- 下一篇: C# 获取鼠标相对当前窗口坐标的方法