淺談async、await關鍵字 => 深談async、await關鍵字

前言

以前寫過有關異步的文章,對這方面一直比較弱,感受仍是不太理解,因而會花點時間去好好學習這一塊,咱們由淺入深,文中如有敘述不穩妥之處,還請批評指正。html

話題

(1)是否是將方法用async關鍵字標識就是異步方法了呢?編程

(2)是否是沒有await關鍵字的存在async就沒有存在的意義了呢?異步

(3)用異步方法的條件是什麼呢,爲何會有這個條件限制?async

(4)只能調用.NET Framework內置的用await標識的Task,可否自定義實現呢?異步編程

(5)在lambda表達式中是否能夠用async和await關鍵字來實現異步呢(即異步lambda表達式)?學習

上述拋出這幾個話題,明白本文主要講述的話題以及須要深刻了解的知識。this

注意:這裏我將參照園友【反骨仔的文章進行進一步解析spa

async關鍵字

例如異步方法是這樣的:線程

        public static async Task<int> asyncMethod()
        {
            return await Task.Run(() => Calculate());
        }

        static int Calculate()
        {
            return 1 + 1;
        }

那要是以下這樣寫呢?code

        public static async Task<int> asyncMethod()
        {
            var task = Task.Run(() => Calculate());
            return task.Result;
        }

那上述這種寫法是否是也是異步方法呢?答案是【NO】,既然不是異步方法爲何要用async關鍵字來進行標識呢?不是很容易被咱們所誤解呢?好了疑問這麼多咱們一一來解惑。

當方法用async標識時,編譯器主要作了什麼呢?

(1)告訴編譯器這個方法裏面可能會用到await關鍵字來標識該方法是異步的,如此以後,編譯器將會在狀態機中編譯此方法。接着該方法執行到await關鍵字時會處於掛起的狀態直到該異步動做完成後才恢復繼續執行方法後面的動做。

(2)告訴編譯器解析出方法的結果到返回類型中,好比說Task或者Task<TResult>,也就是說將返回值存儲到Task中,若是返回值爲void那麼此時應該會將可能出現的異常存儲到上下文中。

當方法用async標識時,是否是全部調用者都將是異步的呢?

當將方法用async標識時且返回值爲void或者Task或者Task<TReuslt>,此時該方法會在當前線程中一直同步執行。用async標識方法並不會影響方法運行完成是不是同步或者異步,相反,它可以將方法劃分紅多塊,有可能有些在異步中運行,以致於這些方法是異步完成的,而劃分異步和同步方法的邊界就是使用await關鍵字。也就是說若是在方法中未用到await關鍵字時則該方法就是一整塊沒有所謂的劃分,會在同步中運行,在同步中完成。

當方法用async標識時,是否會引發方法的調用會被添加到線程池隊列中或者是建立一個新的線程呢?

顯然不是這樣,當用async標識方法時只是顯示告訴編譯器在該方法中await關鍵字可能會被用到,當執行到await關鍵字開始處於掛起的狀態知道異步動做執行完成才恢復(異步操做是在狀態機中【有關狀態機請看這裏:Async和Await異步編程的原理】完成,完成後此時纔會建立一個線程),這也就是爲何在方法中方法用async標識若是沒有用到await關鍵字IDE會發出警告的緣由。

—————————————————————————————————————————————————————————————————

到了這裏咱們能夠得出結論:不管方法是同步仍是異步均可以用async關鍵字來進行標識,由於用async標識只是顯示代表在該方法內可能會用到await關鍵字使其變爲異步方法,並且將該異步方法進行了明確的劃分,只有用了await關鍵字時纔是異步操做,其他一併爲同步操做。

參數爲何不能使用ref和out關鍵字

返回類型必須爲void或者Task或者Task<TResult>和關鍵字的標識以及其餘就再也不敘述,其中有一條是不能使用ref和out關鍵字,你是背書似的銘記了這一條,仍是略加思索了呢?你想過沒有爲什麼不可呢?

咱們知道用ref和out關鍵字不過是爲了在方法裏面改變其值,也就是是當同步完成時咱們指望被ref或者out關鍵字修飾的值會被設置,可是它們可能在異步完成時或者以後纔會被設置達不到咱們預期,因此在異步方法中不能用ref和out關鍵字。

lambda表達式是否能夠異步呢?

返回類型Task參數能夠爲lambda表達式或者匿名方法對象,那直接對lambda表達式異步是否可行?下面咱們來看看

        public static async Task<T2> CallFuncAsync<T1, T2>(T1 t, Func<T1, T2> func)
        {
            return func.Invoke(t);
        }

        public static async Task<string> GetStringAsync(int value)
        {
            return await Task.Run(() => "xpy0928");
        }

        public static async Task MainAsync()
        {
            string value = await CallFuncAsync<int, string>(30, async (s) => await GetStringAsync(s));
        }

編譯後生成以下錯誤:

由上知異步lambda表達式是不行的,猜想是異步lambda表達式不能轉換爲表達式樹,同時咱們看看上述代碼,CallFunAsync此時並未是異步方法,上述咱們已經敘述過,此時是同步運行,既然上述錯誤,而且代碼也有不可取之處咱們接下來一一進行修改。

string value = await CallFuncAsync<int, string>(30, async (s) => await GetStringAsync(s));

修改成:

string value = await CallFuncAsync<int, string>(30, s => GetStringAsync(s).Result);

解決了編譯錯誤,可是未解決CallFuncAsync爲異步運行,咱們將其修改成異步運行。既然await是針對於Task而操做,咱們將CallFuncAsync中的返回參數設置爲Task便可。

        public static async Task<T2> CallFuncAsync<T1, T2>(T1 t, Func<T1, Task<T2>> func)
        {
            return await func.Invoke(t);
        }

則最終調用時咱們直接調用便可。

   string value = await CallFuncAsync(30, GetStringAsync);

此時CallFuncAsync纔算是異步運行。

補充(2016-10-21 23:11)

對於異步表達式有一點其實表述不太正確,其實我一直仍是有點懷疑異步lambda表達式真的不行嗎,此刻我竟然發現這樣是能夠的:

            var task = Task.Run(async () =>
            {
                await Task.Delay(TimeSpan.FromMilliseconds(5000));
            });

如上不正是異步表達式的影子嗎,因而我將上述代碼進行了改寫,以下:

        public static async Task<Action> CallFuncAsync(Action action)
        {
            return action;
        }

        public static async Task<Action> GetStringAsync()
        {
            return () => Console.WriteLine("xpy0928");
        }

        public static async Task MainAsync()
        {
            await CallFuncAsync(async () => await GetStringAsync());
        }

此時編譯經過,說明表述異步表達式並不是必定不能夠,只是對於無參數的lambda表達式才能夠,而對於有參數的lambda表達式如fun則不能執行異步lambda表達式。

至此能夠基本下結論:

異步lambda表達式只對於無參數的lambda表達式 才能夠(固然這也就沒有了什麼意義),而對於有參數的lambda表達式則產生編譯錯誤則不能執行異步(猜想是沒法轉換成對應的表達式樹)。(不知是否嚴謹或者不妥,如有錯誤之處,還望對此理解的更透徹的園友給出批評性意見)。

爲了驗證這一點,咱們來看看無參數的func委託例子,以下:

        static async Task<string> GetTaskAsync()
        {
            await Task.Delay(TimeSpan.FromMilliseconds(5000));
            return "xpy0928";
        }
var task = Task.Run(async () => await GetTaskAsync());

此時無參數的func委託則編譯經過,應該是驗證了上述觀點(仍是有點懷疑我所下的結論)。

await關鍵字

await關鍵字是這樣用的

await Task.Run(() => "xpy0928");

此時背後究竟發生了什麼呢?咱們上述也說過異步動做時在狀態機中完成,當執行到這裏時,編譯器會自動生成代碼來檢測該動做是否已經完成,若是已經完成則繼續同步執行await關鍵字後面的代碼,經過判斷其狀態機狀態若未完成則會掛起一個繼續的委託爲await關鍵字的對象直到完成爲止,調用這個繼續動做的委託從新進入未完成的這樣一個方法。

好比說: await someObject; 編譯器則會生成以下代碼:

private class FooAsyncStateMachine : IAsyncStateMachine 
{ 
    // Member fields for preserving 「locals」 and other necessary state 
    int $state; 
    TaskAwaiter $awaiter; 
    … 
    public void MoveNext() 
    { 
        // Jump table to get back to the right statement upon resumption 
        switch (this.$state) 
        { 
            … 
            case 2: goto Label2; 
            … 
        } 
        … 
        // Expansion of 「await someObject;」 
        this.$awaiter = someObject.GetAwaiter(); 
        if (!this.$awaiter.IsCompleted) 
        { 
            this.$state = 2; 
            this.$awaiter.OnCompleted(MoveNext); 
            return; 
            Label2: 
        } 
        this.$awaiter.GetResult(); 
        … 
    } 
}

此時講到這裏就要涉及到await背後具體的實現,在Task或者Task<TResult>類裏面有這樣一個返回類型爲 TaskAwaiter 的 GetAwaiter 屬性,而TaskAwaiter中有以下屬性:

    public struct TaskAwaiter : ICriticalNotifyCompletion, INotifyCompletion
    {

        public bool IsCompleted { get; }
        public void GetResult();
        public void OnCompleted(Action continuation);
        public void UnsafeOnCompleted(Action continuation);
    }

經過IsComplete來判斷是否已經完成。這個有什麼做用呢?經過看到背後具體實現,咱們能夠本身簡單實現異步擴展方法,當咱們在Task中查看其方法會有這樣的提示:

下面咱們就來實現這樣的效果,給TimeSpan添加異步方法:

    public static class Extend
    {
        public static TaskAwaiter GetAwaiter(this TimeSpan timeSpan)
        {

            return Task.Delay(timeSpan).GetAwaiter();
        }
    }

此時異步方法則是這樣的:

總結 

本節咱們詳細講述了async和await關鍵字的使用和一些基本原理以及解釋其緣由,但願經過對本文的學習,對你們可以更好的去理解異步,我也在學習中,Over。

參考資料

Async/Await FAQ

await anything;

相關文章
相關標籤/搜索