C# 5.0中引入了async 和 await。這兩個關鍵字可以讓你更方便的寫出非同步程式碼。
看個例子:
- public class MyClass
- {
- public MyClass()
- {
- DisplayValue(); //這裡不會阻塞
- System.Diagnostics.Debug.WriteLine("MyClass() End.");
- }
- public Task<double> GetValueAsync(double num1, double num2)
- {
- return Task.Run(() =>
- {
- for (int i = 0; i < 1000000; i++)
- {
- num1 = num1 / num2;
- }
- return num1;
- });
- }
- public async void DisplayValue()
- {
- double result = await GetValueAsync(1234.5, 1.01);//此處會開新執行緒處理GetValueAsync任務,然後方法馬上返回
- //這之後的所有程式碼都會被封裝成委託,在GetValueAsync任務完成時呼叫
- System.Diagnostics.Debug.WriteLine("Value is : " + result);
- }
- }
上面在MyClass的建構函式裡呼叫了async關鍵字標記的非同步方法DisplayValue(),DisplayValue()方法裡執行了一個await關鍵字標記的非同步任務GetValueAsync(),這個非同步任務必須是以Task或者Task<TResult>作為返回值的,而我們也看到,非同步任務執行完成時實際返回的型別是void或者TResult,DisplayValue()方法裡await GetValueAsync()之後的所有程式碼都會在非同步任務完成時才會執行。
DisplayValue()方法實際執行的程式碼如下:
- public void DisplayValue()
- {
- System.Runtime.CompilerServices.TaskAwaiter<double> awaiter = GetValueAsync(1234.5, 1.01).GetAwaiter();
- awaiter.OnCompleted(() =>
- {
- double result = awaiter.GetResult();
- System.Diagnostics.Debug.WriteLine("Value is : " + result);
- });
- }
可以看到,async和await關鍵字只是把上面的程式碼變得更簡單易懂而已。
程式的輸出如下:
MyClass() End.
Value is : 2.47032822920623E-322
以下是我寫的一個靜態類,可以方便將一個普通Function執行非同步呼叫:
- public static class TaskAsyncHelper
- {
- /// <summary>
- /// 將一個方法function非同步執行,在執行完畢時執行回撥callback
- /// </summary>
- /// <param name="function">非同步方法,該方法沒有引數,返回型別必須是void</param>
- /// <param name="callback">非同步方法執行完畢時執行的回撥方法,該方法沒有引數,返回型別必須是void</param>
- public static async void RunAsync(Action function, Action callback)
- {
- Func<System.Threading.Tasks.Task> taskFunc = () =>
- {
- return System.Threading.Tasks.Task.Run(() =>
- {
- function();
- });
- };
- await taskFunc();
- if (callback != null)
- callback();
- }
- /// <summary>
- /// 將一個方法function非同步執行,在執行完畢時執行回撥callback
- /// </summary>
- /// <typeparam name="TResult">非同步方法的返回型別</typeparam>
- /// <param name="function">非同步方法,該方法沒有引數,返回型別必須是TResult</param>
- /// <param name="callback">非同步方法執行完畢時執行的回撥方法,該方法引數為TResult,返回型別必須是void</param>
- public static async void RunAsync<TResult>(Func<TResult> function, Action<TResult> callback)
- {
- Func<System.Threading.Tasks.Task<TResult>> taskFunc = ()=>
- {
- return System.Threading.Tasks.Task.Run(()=>
- {
- return function();
- });
- };
- TResult rlt = await taskFunc();
- if(callback != null)
- callback(rlt);
- }
- }
使用很簡單,將方法名作為引數傳進去就行了,最常用的是把很耗時的序列化函式傳進去,以免阻塞UI程式,造成卡頓現象,影響使用者體驗。