天天看點

C#的async和await

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