天天看點

C#使用委托調用主線程上的控件

摘自其他人部落格,自己試過确實解決問題。(如在自己定義的線程裡面給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;這樣便完成了對其它線程中的控件進行操作

本文轉載自C#使用委托調用主線程上的控件