c#中使用多線程訪問winform中控件的若干問題_第1頁
c#中使用多線程訪問winform中控件的若干問題_第2頁
c#中使用多線程訪問winform中控件的若干問題_第3頁
c#中使用多線程訪問winform中控件的若干問題_第4頁
全文預(yù)覽已結(jié)束

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

1、精選優(yōu)質(zhì)文檔-傾情為你奉上精選優(yōu)質(zhì)文檔-傾情為你奉上專心-專注-專業(yè)專心-專注-專業(yè)精選優(yōu)質(zhì)文檔-傾情為你奉上專心-專注-專業(yè)我們在做winform應(yīng)用的時候,大部分情況下都會碰到使用多線程控制界面上控件信息的問題。然而我們并不能用傳統(tǒng)方法來做這個問題,下面我將詳細的介紹。 首先來看傳統(tǒng)方法:public partial class Form1 : Form public Form1() InitializeComponent(); private void Form1_Load(object sender, EventArgs e) Thread thread = new Thread(Th

2、readFuntion); thread.IsBackground = true; thread.Start(); private void ThreadFuntion() while (true) this.textBox1.Text = DateTime.Now.ToString(); Thread.Sleep(1000); 運行這段代碼,我們會看到系統(tǒng)拋出一個異常:Cross-thread operation not valid:Control textBox1 accessed from a thread other than the thread it was created on

3、. 這是因為.net 2.0以后加強了安全機制,不允許在winform中直接跨線程訪問控件的屬性。那么怎么解決這個問題呢,下面提供幾種方案。 第一種方案,我們在Form1_Load()方法中加一句代碼:private void Form1_Load(object sender, EventArgs e) Control.CheckForIllegalCrossThreadCalls = false; Thread thread = new Thread(ThreadFuntion); thread.IsBackground = true; thread.Start(); 加入這句代碼以后發(fā)現(xiàn)程

4、序可以正常運行了。這句代碼就是說在這個類中我們不檢查跨線程的調(diào)用是否合法(如果沒有加這句話運行也沒有異常,那么說明系統(tǒng)以及默認的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發(fā)現(xiàn)它是一個static的,也就是說無論我們在項目的什么地方修改了這個值,他就會在全局起作用。而且像這種跨線程訪問是否存在異常,我們通常都會去檢查。如果項目中其他人修改了這個屬性,那么我們的方案就失敗了,我們要采取另外的方案。 下面來看第二種方案,就是使用delegate和invoke來從其他線程中控制控件信息。網(wǎng)上有很多人寫了這種控

5、制方式,然而我看了很多這種帖子,表明上看來是沒有什么問題的,但是實際上并沒有解決這個問題,首先來看網(wǎng)絡(luò)上的那種不完善的方式:public partial class Form1 : Form private delegate void FlushClient();/代理 public Form1() InitializeComponent(); private void Form1_Load(object sender, EventArgs e) Thread thread = new Thread(CrossThreadFlush);thread.IsBackground=true; thr

6、ead.Start(); private void CrossThreadFlush() /將代理綁定到方法 FlushClient fc = new FlushClient(ThreadFuntion); this.BeginInvoke(fc);/調(diào)用代理 private void ThreadFuntion() while (true) this.textBox1.Text = DateTime.Now.ToString(); Thread.Sleep(1000); 使用這種方式我們可以看到跨線程訪問的異常沒有了。但是新問題出現(xiàn)了,界面沒有響應(yīng)了。為什么會出現(xiàn)這個問題,我們只是讓新開的線

7、程無限循環(huán)刷新,理論上應(yīng)該不會對主線程產(chǎn)生影響的。其實不然,這種方式其實相當于把這個新開的線程“注入”到了主控制線程中,它取得了主線程的控制。只要這個線程不返回,那么主線程將永遠都無法響應(yīng)。就算新開的線程中不使用無限循環(huán),使可以返回了。這種方式的使用多線程也失去了它本來的意義。 現(xiàn)在來讓我們看看推薦的解決方案:public partial class Form1 : Form private delegate void FlushClient();/代理 public Form1() InitializeComponent(); private void Form1_Load(object s

8、ender, EventArgs e) Thread thread = new Thread(CrossThreadFlush); thread.IsBackground = true; thread.Start(); private void CrossThreadFlush() while (true) /將sleep和無限循環(huán)放在等待異步的外面 Thread.Sleep(1000); ThreadFunction(); private void ThreadFunction() if (this.textBox1.InvokeRequired)/等待異步 FlushClient fc = new FlushClient(ThreadFunction); this.Invoke(fc);/通過代理調(diào)用刷新方法 else this

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論