由於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
總感覺MSDN上的翻譯讓人無法一看就明白,可能是自己智力不夠吧~~
最後就是我們的具體程序了:
delegate void
aa(string s);//創建一個代理
private void pri(string
t)//這個就是我們的函數,我們把要對控件進行的操作放在這裏
{
if
(!richTextBox1.InvokeRequired)//判斷是否需要進行喚醒的請求,如果控件與主線程在一個線程內,可以寫成
if(!InvokeRequired)
{
MessageBox.Show("同一線程內");
richTextBox1.Text =t;
}
else
{
MessageBox.Show("不是同一個線程");
aa a1 =new aa(pri);
Invoke(a1,new object []{t});//執行喚醒操作
}
}
private void Form1_Load(object
sender, System.EventArgse)
{
Thread
newthread = new Thread(new ThreadStart(ttread));
newthread.Start();
}
void ttread()
{
pri("sdfs");
}
執行結果先調出一個提示框顯示“不是同一個線程”,然後跳出提示框顯示“同一線程內”,然後richTextBox1中的text值爲sdfs;這樣便完成了對其它線程中的控件進行操作。