C#異步調(diào)用四大方法詳解
C#異步調(diào)用四大方法是什么呢?C#異步調(diào)用四大方法的使用是如何進(jìn)行的呢?讓我們首先了解下什么時(shí)候用到C#異步調(diào)用:
.NET Framework 允許您C#異步調(diào)用任何方法。定義與您需要調(diào)用的方法具有相同簽名的委托;公共語(yǔ)言運(yùn)行庫(kù)將自動(dòng)為該委托定義具有適當(dāng)簽名的 BeginInvoke 和 EndInvoke 方法。
BeginInvoke 方法用于啟動(dòng)C#異步調(diào)用。它與您需要異步執(zhí)行的方法具有相同的參數(shù),只不過(guò)還有兩個(gè)額外的參數(shù)(將在稍后描述)。BeginInvoke 立即返回,不等待C#異步調(diào)用完成。BeginInvoke 返回 IasyncResult,可用于監(jiān)視調(diào)用進(jìn)度。
EndInvoke 方法用于檢索C#異步調(diào)用結(jié)果。調(diào)用 BeginInvoke 后可隨時(shí)調(diào)用 EndInvoke 方法;如果C#異步調(diào)用未完成,EndInvoke 將一直阻塞到C#異步調(diào)用完成。EndInvoke 的參數(shù)包括您需要異步執(zhí)行的方法的 out 和 ref 參數(shù)(在 Visual Basic 中為
注意 Visual Studio .NET 中的智能感知功能會(huì)顯示 BeginInvoke 和 EndInvoke 的參數(shù)。如果您沒(méi)有使用 Visual Studio 或類似的工具,或者您使用的是 C# 和 Visual Studio .NET,請(qǐng)參見(jiàn)異步方法簽名獲取有關(guān)運(yùn)行庫(kù)為這些方法定義的參數(shù)的描述。
本主題中的代碼演示了四種使用 BeginInvoke 和 EndInvoke 進(jìn)行C#異步調(diào)用的常用方法。調(diào)用了 BeginInvoke 后,可以:
· 進(jìn)行某些操作,然后調(diào)用 EndInvoke 一直阻塞到調(diào)用完成。
· 使用 IAsyncResult.AsyncWaitHandle 獲取 WaitHandle,使用它的 WaitOne 方法將執(zhí)行一直阻塞到發(fā)出 WaitHandle 信號(hào),然后調(diào)用 EndInvoke。
· 輪詢由 BeginInvoke 返回的 IAsyncResult,確定C#異步調(diào)用何時(shí)完成,然后調(diào)用 EndInvoke。
· 將用于回調(diào)方法的委托傳遞給 BeginInvoke。該方法在C#異步調(diào)用完成后在 ThreadPool 線程上執(zhí)行,它可以調(diào)用 EndInvoke。
警告:始終在C#異步調(diào)用完成后調(diào)用 EndInvoke。
測(cè)試方法和異步委托
四個(gè)示例全部使用同一個(gè)長(zhǎng)期運(yùn)行的測(cè)試方法 TestMethod。該方法顯示一個(gè)表明它已開(kāi)始處理的控制臺(tái)信息,休眠幾秒鐘,然后結(jié)束。TestMethod 有一個(gè) out 參數(shù)(在 Visual Basic 中為
下面的代碼示例顯示 TestMethod 以及代表 TestMethod 的委托;若要使用任一示例,請(qǐng)將示例代碼追加到這段代碼中。
注意 為了簡(jiǎn)化這些示例,TestMethod 在獨(dú)立于 Main() 的類中聲明?;蛘撸琓estMethod 可以是包含 Main() 的同一類中的 static 方法(在 Visual Basic 中為 Shared)。
- using System;
- using System.Threading;
- public class AsyncDemo {
- // The method to be executed asynchronously.
- //
- public string TestMethod(
- int callDuration, out int threadId) {
- Console.WriteLine("Test method begins.");
- Thread.Sleep(callDuration);
- threadId = AppDomain.GetCurrentThreadId();
- return "MyCallTime was " + callDuration.ToString();
- }
- }
- // The delegate must have the same signature as the method
- // you want to call asynchronously.
- public delegate string AsyncDelegate(
- int callDuration, out int threadId);
- using System;
- using System.Threading;
- public class AsyncDemo {
- // The method to be executed asynchronously.
- //
- public string TestMethod(
- int callDuration, out int threadId) {
- Console.WriteLine("Test method begins.");
- Thread.Sleep(callDuration);
- threadId = AppDomain.GetCurrentThreadId();
- return "MyCallTime was " + callDuration.ToString();
- }
- }
- // The delegate must have the same signature as the method
- // you want to call asynchronously.
- public delegate string AsyncDelegate(
- int callDuration, out int threadId);
C#異步調(diào)用四大方法之使用 EndInvoke 等待異步調(diào)用
異步執(zhí)行方法的最簡(jiǎn)單方式是以 BeginInvoke 開(kāi)始,對(duì)主線程執(zhí)行一些操作,然后調(diào)用 EndInvoke。EndInvoke 直到C#異步調(diào)用完成后才返回。這種技術(shù)非常適合文件或網(wǎng)絡(luò)操作,但是由于它阻塞 EndInvoke,所以不要從用戶界面的服務(wù)線程中使用它。
- public class AsyncMain {
- static void Main(string[] args) {
- // The asynchronous method puts the thread id here.
- int threadId;
- // Create an instance of the test class.
- AsyncDemo ad = new AsyncDemo();
- // Create the delegate.
- AsyncDelegate dlgt = new AsyncDelegate(ad.TestMethod);
- // Initiate the asychronous call.
- IAsyncResult ar = dlgt.BeginInvoke(3000,
- out threadId, null, null);
- Thread.Sleep(0);
- Console.WriteLine("Main thread {0} does some work.",
- AppDomain.GetCurrentThreadId());
- // Call EndInvoke to Wait for
- //the asynchronous call to complete,
- // and to retrieve the results.
- string ret = dlgt.EndInvoke(out threadId, ar);
- Console.WriteLine("The call executed on thread {0},
- with return value \"{1}\".", threadId, ret);
- }
- }
C#異步調(diào)用四大方法之使用 WaitHandle 等待異步調(diào)用
等待 WaitHandle 是一項(xiàng)常用的線程同步技術(shù)。您可以使用由 BeginInvoke 返回的 IAsyncResult 的 AsyncWaitHandle 屬性來(lái)獲取 WaitHandle。C#異步調(diào)用完成時(shí)會(huì)發(fā)出 WaitHandle 信號(hào),而您可以通過(guò)調(diào)用它的 WaitOne 等待它。
如果您使用 WaitHandle,則在C#異步調(diào)用完成之后,但在通過(guò)調(diào)用 EndInvoke 檢索結(jié)果之前,可以執(zhí)行其他處理。
- public class AsyncMain {
- static void Main(string[] args) {
- // The asynchronous method puts the thread id here.
- int threadId;
- // Create an instance of the test class.
- AsyncDemo ad = new AsyncDemo();
- // Create the delegate.
- AsyncDelegate dlgt = new AsyncDelegate(ad.TestMethod);
- // Initiate the asychronous call.
- IAsyncResult ar = dlgt.BeginInvoke(3000,
- out threadId, null, null);
- Thread.Sleep(0);
- Console.WriteLine("Main thread {0} does some work.",
- AppDomain.GetCurrentThreadId());
- // Wait for the WaitHandle to become signaled.
- ar.AsyncWaitHandle.WaitOne();
- // Perform additional processing here.
- // Call EndInvoke to retrieve the results.
- string ret = dlgt.EndInvoke(out threadId, ar);
- Console.WriteLine("The call executed on thread {0},
- with return value \"{1}\".", threadId, ret);
- }
- }
C#異步調(diào)用四大方法之輪詢異步調(diào)用完成
您可以使用由 BeginInvoke 返回的 IAsyncResult 的 IsCompleted 屬性來(lái)發(fā)現(xiàn)C#異步調(diào)用何時(shí)完成。從用戶界面的服務(wù)線程中進(jìn)行C#異步調(diào)用時(shí)可以執(zhí)行此操作。輪詢完成允許用戶界面線程繼續(xù)處理用戶輸入。
- public class AsyncMain {
- static void Main(string[] args) {
- // The asynchronous method puts the thread id here.
- int threadId;
- // Create an instance of the test class.
- AsyncDemo ad = new AsyncDemo();
- // Create the delegate.
- AsyncDelegate dlgt = new AsyncDelegate(ad.TestMethod);
- // Initiate the asychronous call.
- IAsyncResult ar = dlgt.BeginInvoke(3000,
- out threadId, null, null);
- // Poll while simulating work.
- while(ar.IsCompleted == false) {
- Thread.Sleep(10);
- }
- // Call EndInvoke to retrieve the results.
- string ret = dlgt.EndInvoke(out threadId, ar);
- Console.WriteLine("The call executed on thread {0},
- with return value \"{1}\".", threadId, ret);
- }
- }
C#異步調(diào)用四大方法之異步調(diào)用完成時(shí)執(zhí)行回調(diào)方法
如果啟動(dòng)異步調(diào)用的線程不需要處理調(diào)用結(jié)果,則可以在調(diào)用完成時(shí)執(zhí)行回調(diào)方法。回調(diào)方法在 ThreadPool 線程上執(zhí)行。
要使用回調(diào)方法,必須將代表該方法的 AsyncCallback 委托傳遞給 BeginInvoke。也可以傳遞包含回調(diào)方法將要使用的信息的對(duì)象。例如,可以傳遞啟動(dòng)調(diào)用時(shí)曾使用的委托,以便回調(diào)方法能夠調(diào)用 EndInvoke。
- public class AsyncMain {
- // Asynchronous method puts the thread id here.
- private static int threadId;
- static void Main(string[] args) {
- // Create an instance of the test class.
- AsyncDemo ad = new AsyncDemo();
- // Create the delegate.
- AsyncDelegate dlgt = new AsyncDelegate(ad.TestMethod);
- // Initiate the asychronous call. Include an AsyncCallback
- // delegate representing the callback method, and the data
- // needed to call EndInvoke.
- IAsyncResult ar = dlgt.BeginInvoke(3000,
- out threadId,
- new AsyncCallback(CallbackMethod),
- dlgt );
- Console.WriteLine("Press Enter to close application.");
- Console.ReadLine();
- }
- // Callback method must have the same signature as the
- // AsyncCallback delegate.
- static void CallbackMethod(IAsyncResult ar) {
- // Retrieve the delegate.
- AsyncDelegate dlgt = (AsyncDelegate) ar.AsyncState;
- // Call EndInvoke to retrieve the results.
- string ret = dlgt.EndInvoke(out threadId, ar);
- Console.WriteLine("The call executed on thread {0},
- with return value \"{1}\".", threadId, ret);
- }
- }
C#異步調(diào)用四大方法的基本內(nèi)容就向你介紹到這里,希望對(duì)你了解和學(xué)習(xí)C#異步調(diào)用有所幫助。
【編輯推薦】