C#中的invoke和begininvoke

原帖:http://www.cnblogs.com/Z-King/archive/2011/11/03/2234337.html

一直對invoke和begininvoke的使用和概念比較混亂,這兩天看了些資料,對這兩個的用法和原理有了些新的認識和理解。

 首先說下,invoke和begininvoke的使用有兩種情況:

  1. control中的invoke、begininvoke。

  2. delegrate中的invoke、begininvoke。  

  這兩種情況是不同的,我們這裏要講的是第1種。下面我們在來說下.NET中對invoke和begininvoke的官方定義。

  control.invoke(參數delegate)方法:在擁有此控件的基礎窗口句柄的線程上執行指定的委託。

  control.begininvoke(參數delegate)方法:在創建控件的基礎句柄所在線程上異步執行指定委託。

  根據這兩個概念我們大致理解invoke表是同步、begininvoke表示異步。但是如何來進行同步和異步呢?我們來做一個測試。

invoke 例子:

 private void button1_Click(object sender, EventArgs e)
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"AAA");
            invokeThread = new Thread(new ThreadStart(StartMethod));
            invokeThread.Start();
            string a = string.Empty;
            for (int i = 0; i < 3; i++)      //調整循環次數,看的會更清楚
            {
                Thread.Sleep(1000);   
                a = a + "B";
            }
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+a);
}

 private void StartMethod()
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"CCC");
            button1.Invoke(new invokeDelegate(invokeMethod));  
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"DDD");
}

 private void invokeMethod()
{
            //Thread.Sleep(3000);
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString() + "EEE");

結論:我們運行後,看下程序的運行順序,1AAA->3CCC和1BBB->1EEE ->3DDD 。 

解釋:主線程運行1AAA,然後1BBB和子線程3CCC同時執行,然後通過invoke來將invokemethod方法提交給主線程,然後子線程等待主線程執行,直到主線程將invokemethod方法執行完成(期間必須等待主線程的任務執行完成,纔會去執行invoke提交的任務),最後執行子線程3DDD。

 

begininvoke 例子:

private void button1_Click(object sender, EventArgs e)
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"AAA");
            invokeThread = new Thread(new ThreadStart(StartMethod));
            invokeThread.Start();
            string a = string.Empty;
            for (int i = 0; i < 3; i++)      //調整循環次數,看的會更清楚
            {
                Thread.Sleep(1000);   
                a = a + "B";
            }
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+a);
}

 private void StartMethod()
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"CCC");
            button1.BeginInvoke(new invokeDelegate(invokeMethod));  
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"DDD");
}

 private void beginInvokeMethod()
        {
            //Thread.Sleep(3000);
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString() + "EEEEEEEEEEEE");
        }

 

結論: 我們運行後看看執行的結果:1AAA->1BBB和3CCC->1EEE和3DDD。

解釋: 主線程運行1AAA,然後1BBB和子線程3CCC同時執行,然後通過begininvoke來將invokemethod方法提交給主線程,然後主線程執行1EEE(主線程自己的任務執行完成), 同時子線程繼續執行3DDD。

 

通過這個兩段代碼的測試比較,我們會發現其實invoke和begininvoke所提交的委託方法都是在主線程中執行的,其實根據我invoke和begininvoke的定義我們要在子線程中來看這個問題,在invoke例子中我們會發現invoke所提交的委託方法執行完成後,才能繼續執行DDD;在begininvoke例子中我們會發現begininvoke所提交的委託方法後,子線程講繼續執行DDD,不需要等待委託方法的完成。 那麼現在我們在回想下invoke(同步)和begininvoke(異步)的概念,其實它們所說的意思是相對於子線程而言的,其實對於控件的調用總是由主線程來執行的。我們很多人搞不清這個同步和異步,主要還是因爲我們把參照物選錯了。其實有時候光看概念是很容易理解錯誤的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章