源自:http://blog.csdn.net/wyvbboy/article/details/50587617 摘自其餘人博客,本身試過確實解決問題。(如在本身定義的線程裏面給textbox賦值) 因爲Windows窗體控件本質上不是線程安全的。所以若是有兩個或多個線程適度操做某一控件的狀態(set value),則可能會迫使該控件進入一種不一致的狀態。還可能出現其餘與線程相關的bug,包括爭用和死鎖的狀況。因此VS2005這一改動即可以加強 線程安全性。 我想你們更關心的是如何解決這個問題,如何才能操做其它線程中的控件而不引起異常,接下來咱們就來探討下這個問題: 第一種方法: 這種方法我沒用過,由於你們推薦不要使用,因此我沒去實驗過,具體方法以下(摘自網上): 設置System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls=false;(winform.下)若是在你的程序初始化的時候設置了這個屬性,並且在你的控件中使用的都是微軟Framework類庫中的控件的話,系統就不會再拋出你上面所說的這個錯誤了。固然這只是爲了將VS2003的代碼轉換到VS2005下所使用的一種常見的方法。不建議採用; 第二種方法,也是我今天主要要講的就是利用delegate和invoke這個方法: 思路:把想對另外一線程中的控件實施的操做放到一個函數中,而後使用delegate代理那個函數,而且在那個函數中加入一個判斷,用 InvokeRequired來判斷調用這個函數的線程是否和控件線程在同一線程中,若是是則直接執行對控件的操做,不然利用控件的Invoke或 BeginInvoke方法來執行這個代理。 在繼續講解下去以前咱們先來看一下這裏提到的幾個方法(若是對如下兩個東東已經瞭解了就能夠跳過) 首先是Invoke Invoke的中文解釋是喚醒,它有兩種參數類型咱們這裏只講一種即(Delegate, Object[]) Delegate就是前面提到的那個代理,而Object[]則是用來存放Delegate所代理函數的參數 MSDN上關於INVOKE方法有以下說明:在擁有控件的基礎窗口句柄的線程上,用指定的參數列表執行指定委託。 用通俗的話講就是利用控件的INVOKE方法,使該控件所在的線程執行這個代理,也就是執行咱們想對控件進行的操做,至關於喚醒了這個操做; 其次是控件的InvokeRequired這個屬性 MSDN上關於它的解釋是獲取一個值,該值指示調用方在對控件進行方法調用時是否必須調用Invoke方法,由於調用方位於建立控件所在的線程之外的線程中。 有通俗的話講就是返回一個值,若是與控件屬於同一個線程,則不須要進行喚醒的請求,也就是返回值爲False,不然則須要進行喚醒的請求,返回爲 true 最後就是咱們的具體程序了: delegate void aa(string s);//建立一個代理安全
private void pri(string t)//這個就是咱們的函數,咱們把要對控件進行的操做放在這裏 { if(!richTextBox1.InvokeRequired)//判斷是否須要進行喚醒的請求,若是控件與主線程在一個線程內,能夠寫成 if(!InvokeRequired) { MessageBox.Show("同一線程內"); richTextBox1.Text =t; return; } if(richTextBox1.InvokeRequired) { MessageBox.Show("不是同一個線程"); aa a1 =new aa(pri); Invoke(a1,new object []{t});//執行喚醒操做 return; } } private void Form1_Load(object sender, System.EventArgs e) { Thread newthread = new Thread(new ThreadStart(ttread)); newthread.Start(); } void ttread() { pri("sdfs"); }
執行結果先調出一個提示框顯示「不是同一個線程」,而後跳出提示框顯示「同一線程內」,而後richTextBox1中的text值爲sdfs;這樣便完成了對其它線程中的控件進行操做函數