文章出處

怎么使用異步,就是用委托進行處理,如果委托對象在調用列表中只有一個方法,它就可以異步執行這個方法。委托類有兩個方法,叫做BeginInvoke和EndInvoke,它們是用來異步執行使用。

異步有三種模式

  1. 等待模式,在發起了異步方法以及做了一些其它處理之后,原始線程就中斷,并且等待異步方法完成之后再繼續。
  2. 輪詢模式,原始線程定期檢查發起的線程是否完成,如果沒有則可以繼續做一些其它的事情。
  3. 回調模式,原始線程一直在執行,無需等待或檢查發起的線程是否完成。在發起的線程中的引用方法完成之后,發起的線程就會調用回調方法,由回調方法在調用EndInvoke之前處理異步方法的結構。

在學習異步編程之前,先看看BeginInvoke和EndInvoke方法。

BeginInvoke方法

  1. 在調用BeginInvoke時,參數列表中的實參組成如下:

    1)  引用方法需要的參數。

    2)  兩個額外的參數——callback參數和state參數。

  1. BeginInvoke從線程池中獲取一個線程并且在新的線程開始時運行引用方法。
  2. BeginInvoke返回給調用線程一個實現IasyncResult接口的對象。這個接口引用包含了異步方法的當前狀態,原始線程然后可以繼續執行。

EndInvoke方法

  1. 它接受一個由BeginInvoke方法返回的IasyncResult對象的引用,并找到它關聯的線程。
  2. 如果線程池的線程已經退出,EndInvoke做如下的事情。

    1)  它清理退出線程的狀態并且釋放它的資源。

    2)  它找到引用方法返回的值并且把它的值作為返回值。

  1. 如果當EndInvoke被調用時線程池的線程仍然在運行,調用線程就會停止并等待,直到清理完畢并返回值。因為EndInvoke是為開啟的線程進行清理,所以必須確保對每一個BeginInvoke都調用EndInvoke。
  2. 如果異步方法觸發了異常,在調用EndInvoke時會拋出異常。

等待模式

在這種模式里,原始線程發起一個異步方法的調用,做一些其它處理,然后停止并等待,直到開啟的線程結束。如下圖

 

 

 

 

這段代碼產生了如下輸出。

 

 

  既然我們已經看到了BeginInvoke和EndInoke的最簡單形式,可以進一步了解IasyncResult了,它是使用這些方法的必要部分。

BeginInvoke返回一個IasyncResult接口的引用(內部是AsyncResult類的對象)。AsyncResult類表現了異步方法的狀態。如下圖:

 

  1. 當我們調用委托對象的BeginInvoke方法時,系統創建了一個AsyncResult類的對象。然而,它不返回類的對象的引用,而是返回對象中包含的IasyncResult接口的引用。
  2. AsyncResult對象包含一個叫做AsyncDelegate的屬性,它返回一個指向被調用來開啟異步方法的委托的引用。但是這個屬性是類對象的一部分而是接口的一部分。
  3. IsCompleted屬性返回一個布爾值,表示異步方法是否完成。
  4. AsyncState屬性返回一個對象的引用,它被作為BeginInvoke方法調用時的state參數。它返回object類型的引用,稍后再講解。。

輪詢模式

  在輪詢模式中,原始線程發起了異步方法的調用,做一些其它處理,然后使用IAsyncResult對象的IsCompleted屬性來定期檢查開啟的線程是否完成。如果異步方法已經完成,原始線程就調用EndInvoke并繼續。否則,它做一些其它處理,然后過一會兒再檢查。如下圖:

 

這段代碼產生了如下輸出。

 

 

回調模式

  在之前的等待模式與輪詢模式中,初始線程繼續它自己的控制流程,直到它知道開啟的線程完成。然后,它獲取結果并繼續。

  回調模式的不同之處在于,一旦初始線程發起了異步方法,它就自己管自己了,不再考慮同步。當異步方法調用結束之后,系統調用一個用戶自定義的方法來處理結束,并且調用委托的EndInvoke方法。這個用戶自定義的方法叫做回調方法或回調。

  BeginInvoke的參數列表中最后的兩個額外參數被回調方法用做:

    1)  第一個參數,callback參數,是回調方法的名字。

    2)  第二個參數,state參數,可以是null或要傳入回調方法的一個對象數據。我們可以通過使用IAsyncResult參數的AsyncState屬性來獲取這個對象。參數類型是object

  1. 回調方法的簽名和返回類型必須和AsyncCallback委托類型所描述的形式一致。它需要方法接受一個IAsyncResult作為參數并且返回類型是void。如下所示:

Void AsyncCallback(IAsyncResult iar)

  1. 在回調方法內,我們的代碼應該調用委托的EndInvoke方法來處理異步方法執行后的輸出值。要調用委托的EndInvoke方法,我們肯定需要委托對象的引用,而它在初始線程中,不在開啟線程中。如果我們不使用BeginInvoke的state參數作其它的用途,可以使用它發送委托的引用給回調方法。否則,我們可以從發送給方法作為參數的IAsyncResult對象中提取出委托的引用。

    1)  給回調方法的參數只有一個,就是剛結束的異步方法的IAsyncResult接口的引用,要記住,IAsyncResult接口對象在AsyncResult類對象的內部。

    2)  盡管IAsyncResult接口沒有委托對象的引用,而包含它的AsyncResult類對象卻有委托對象的引用。

    3)  有了類對象的引用,我們現在就可以調用類對象的AsyncDelegate屬性并且把它轉化為合適的委托類型。這樣就得到了委托引用,我們可以用它來調用EndInvoke。

如下代碼所示:

 

 

 

 

這段代碼產生了如下輸出。

 

那么以上的異步內容已經基本講解完畢。

說明:異步編程在.Net4.0中,有了更好的方式處理,簡化了編程的復雜度,使用Task類(在System.Threading.Tasks命名空間中)處理,但原理是一樣的。

原文地址:http://www.cnblogs.com/zwt-blog/p/4812530.html#3611906


文章列表


不含病毒。www.avast.com
arrow
arrow
    全站熱搜
    創作者介紹
    創作者 大師兄 的頭像
    大師兄

    IT工程師數位筆記本

    大師兄 發表在 痞客邦 留言(0) 人氣()