[轉]說說C#的async和await

RobotTech發表於2016-07-07

C# 5.0中引入了async 和 await。這兩個關鍵字可以讓你更方便的寫出非同步程式碼。

看個例子:

 

 
  1. public class MyClass  
  2. {  
  3.     public MyClass()  
  4.     {  
  5.         DisplayValue(); //這裡不會阻塞  
  6.         System.Diagnostics.Debug.WriteLine("MyClass() End.");  
  7.     }  
  8.     public Task<double> GetValueAsync(double num1, double num2)  
  9.     {  
  10.         return Task.Run(() =>  
  11.         {  
  12.             for (int i = 0; i < 1000000; i++)  
  13.             {  
  14.                 num1 = num1 / num2;  
  15.             }  
  16.             return num1;  
  17.         });  
  18.     }  
  19.     public async void DisplayValue()  
  20.     {  
  21.         double result = await GetValueAsync(1234.5, 1.01);//此處會開新執行緒處理GetValueAsync任務,然後方法馬上返回  
  22.         //這之後的所有程式碼都會被封裝成委託,在GetValueAsync任務完成時呼叫  
  23.         System.Diagnostics.Debug.WriteLine("Value is : " + result);  
  24.     }  
  25. }  

 

上面在MyClass的建構函式裡呼叫了async關鍵字標記的非同步方法DisplayValue(),DisplayValue()方法裡執行了一個await關鍵字標記的非同步任務GetValueAsync(),這個非同步任務必須是以Task或者Task<TResult>作為返回值的,而我們也看到,非同步任務執行完成時實際返回的型別是void或者TResult,DisplayValue()方法裡await GetValueAsync()之後的所有程式碼都會在非同步任務完成時才會執行。

DisplayValue()方法實際執行的程式碼如下:

 

  1. public void DisplayValue()  
  2. {  
  3.     System.Runtime.CompilerServices.TaskAwaiter<double> awaiter = GetValueAsync(1234.5, 1.01).GetAwaiter();  
  4.     awaiter.OnCompleted(() =>  
  5.         {  
  6.             double result = awaiter.GetResult();  
  7.             System.Diagnostics.Debug.WriteLine("Value is : " + result);  
  8.         });  
  9. }   


可以看到,async和await關鍵字只是把上面的程式碼變得更簡單易懂而已。

 

程式的輸出如下:

MyClass() End.

Value is : 2.47032822920623E-322

 

以下是我寫的一個靜態類,可以方便將一個普通Function執行非同步呼叫:

 

 
  1. public static class TaskAsyncHelper  
  2. {  
  3.     /// <summary>  
  4.     /// 將一個方法function非同步執行,在執行完畢時執行回撥callback  
  5.     /// </summary>  
  6.     /// <param name="function">非同步方法,該方法沒有引數,返回型別必須是void</param>  
  7.     /// <param name="callback">非同步方法執行完畢時執行的回撥方法,該方法沒有引數,返回型別必須是void</param>  
  8.     public static async void RunAsync(Action function, Action callback)  
  9.     {  
  10.         Func<System.Threading.Tasks.Task> taskFunc = () =>  
  11.         {  
  12.             return System.Threading.Tasks.Task.Run(() =>  
  13.             {  
  14.                 function();  
  15.             });  
  16.         };  
  17.         await taskFunc();  
  18.         if (callback != null)  
  19.             callback();  
  20.     }  
  21.   
  22.     /// <summary>  
  23.     /// 將一個方法function非同步執行,在執行完畢時執行回撥callback  
  24.     /// </summary>  
  25.     /// <typeparam name="TResult">非同步方法的返回型別</typeparam>  
  26.     /// <param name="function">非同步方法,該方法沒有引數,返回型別必須是TResult</param>  
  27.     /// <param name="callback">非同步方法執行完畢時執行的回撥方法,該方法引數為TResult,返回型別必須是void</param>  
  28.     public static async void RunAsync<TResult>(Func<TResult> function, Action<TResult> callback)  
  29.     {  
  30.         Func<System.Threading.Tasks.Task<TResult>> taskFunc = ()=>  
  31.             {  
  32.                 return System.Threading.Tasks.Task.Run(()=>  
  33.                     {  
  34.                         return function();  
  35.                     });  
  36.             };  
  37.         TResult rlt = await taskFunc();  
  38.         if(callback != null)  
  39.             callback(rlt);  
  40.     }  
  41. }  


使用很簡單,將方法名作為引數傳進去就行了,最常用的是把很耗時的序列化函式傳進去,以免阻塞UI程式,造成卡頓現象,影響使用者體驗。

相關文章