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);
}
}