重新認識 async/await 語法糖

来源:https://www.cnblogs.com/bobsprite/archive/2019/07/19/11211549.html
-Advertisement-
Play Games

提起.Net中的 async/await,相信很多.neter 第一反應都會是非同步編程,其本質是語法糖,但繼續追查下去,既然是語法糖,那麼經過編譯之後,真正的代碼是什麼樣的,如何執行的?帶著這些疑問,通過網上資料的查詢,可以瞭解到編譯之後,是通過實現 IAsyncStateMachine 的一個狀態 ...


提起.Net中的 async/await,相信很多.neter 第一反應都會是非同步編程,其本質是語法糖,但繼續追查下去,既然是語法糖,那麼經過編譯之後,真正的代碼是什麼樣的,如何執行的?帶著這些疑問,通過網上資料的查詢,可以瞭解到編譯之後,是通過實現 IAsyncStateMachine 的一個狀態機來實現的,博客園裡大神Jeffcky 已經說得很清楚了,傳送門: https://www.cnblogs.com/CreateMyself/p/5983208.html

上述知識對我們理解 async/await 非常重要,但不是本文討論的側重點,觸發筆者寫這篇文章的初衷是:

1.只有Task可以被await嗎,await之後就一定是非同步執行嗎?

答案當然不是,google了一圈後發現,當一個類可以被await,必須滿足以下條件:

a.它必須包含 GetAwaiter() 方法(實例方法或者擴展方法) // 手動劃重點:擴展方法,聰明的你是不是立馬有些思想火花
b.GetAwaiter() 返回awatier實例,並且這個實例包含如下條件:

  • 必須實現 INotifyCompletion 或者 ICriticalNotifyCompletion 介面
  • 必須包含 IsCompleted 公共屬性
  • 必須包含 GetResult() 方法,返回void或者其他返回值

上述條件中INotifyCompletion 介面信息如下:

    //
    // 摘要:
    //     Represents an operation that schedules continuations when it completes.
    public interface INotifyCompletion
    {
        //
        // 摘要:
        //     Schedules the continuation action that's invoked when the instance completes.
        //
        // 參數:
        //   continuation:
        //     The action to invoke when the operation completes.
        //
        // 異常:
        //   T:System.ArgumentNullException:
        //     The continuation argument is null (Nothing in Visual Basic).
        void OnCompleted(Action continuation);
    }

重點上述對於參數 continuation 的解釋:委托在操作完成之後調用。此處遺留一個問題:在誰的操作完成之後調用,是怎麼調用的?

先把上述問題放一邊,我們來自己寫一個可以被await的類,並且觀察前後執行的順序以及是否存線上程切換:

 public class Program {
        static async Task Main (string[] args) {
            Console.WriteLine ($"Begin awati,thread id is {Thread.CurrentThread.ManagedThreadId}");
            int result = await new CustomAwaitable ();
            Console.WriteLine ($"End await,result is {result},thread id is {Thread.CurrentThread.ManagedThreadId}");
            await Task.Delay (Timeout.Infinite);
        }
    }

    public class CustomAwaitable : INotifyCompletion {
        public void OnCompleted (Action continuation) {
            Console.WriteLine ($"Invoke continuation action on completed,thread id is {Thread.CurrentThread.ManagedThreadId}");
            continuation?.Invoke ();
        }

        public int GetResult () {
            Console.WriteLine ($"Get result,thread id is {Thread.CurrentThread.ManagedThreadId}");
            return 100;
        }

        public bool IsCompleted { get; set; }

        public CustomAwaitable GetAwaiter(){
            return this;
        }
    }

上述代碼中,CustomAwaitable 實例滿足了可被await的所有條件,並且正常通過編譯,運行後發現結果如下:

PS D:\git\awaitable\src> dotnet run
Begin main,thread id is 1
Get awatier,thread id is 1
Begin Invoke continuation action on completed,thread id is 1
Get result,thread id is 1
End main,result is 100,thread id is 1
End Invoke

根據上述日誌,可以看出:

  1. 執行前後線程並未發生切換,所以當我們不假思索的回答 await/async 就是非同步編程時,至少是一個不太嚴謹的答案
  2. 最後執行日誌 "End Invoke" 表明:continuation action 這個委托,根據上述調用日誌順序可以大致理解為:編譯器將await之後的代碼封裝為這個 action,在實例完成後調用OnCompleted方法執行了await 之後的代碼(註:實際情況比較複雜,如果有多行await,會轉換為一個狀態機,具體參看文章開頭給出的連接)。

2.瞭解了上述知識之後,那麼我們常規所說的await Task非同步編程又是怎麼回事呢?

  1. 先來看Task部分源碼(傳送門):

上述紅框代碼顯示,Task在GetAwaiter中創建了 TaskAwaiter對象,並將this傳遞。

  1. 再來看TaskAwaiter源碼(傳送門):

看到此處,有了前面的知識,我們會對await task有了更加深入的理解:
Task通過增加一個GetAwatier()函數,同時將自身傳遞給TaskAwaiter類來實現了await語法糖的支持,同時在執行時,調用GetResult()函數的本質是通過 Task.Wait等待非同步線程的執行完成,然後通過回調進行後續的操作。

總結

本文主要對 async/await 語法糖進行分析驗證,同時通過對Task源碼分析,更加深入的理解此語法糖本身的語法,相信通過通過此文,對大家從多個角度去理解非同步編程有幫助,我自己也在不停的學習。

本文代碼示例地址:https://github.com/xBoo/awaitable


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 多數耗時操作可以非同步執行,推薦async/await。 但和UI相關的部分僅能在UI線程執行,這時UI線程的耗時操作,導致界面卡死,不夠友好。 我們可以創建一個單獨的UI線程顯示一個正在載入的視窗,可以在視窗顯示一些動畫之類的。 如果是WinForms那麼很容易做到。 WPF沒有提供這樣簡單的方式, ...
  • 根據市場需求,基於.NET CORE平臺開發的RoadFlow工作流平臺需要實現多語言版本。經過方案討論和比對,決定採用.NET自帶的本地化功能來實現多語言。話不多說,直接上實現方式。 首先修改Startup.cs 在public void ConfigureServices(IServiceCol ...
  • 首次使用C#編寫與COM口有關的程式,期間遇到了很多問題,寫下自己的經驗總結,如有錯漏,歡迎批評指正! 1、新建一個串口類( SerialPort類) 2、串口埠號搜索: 3、讀數據、顯示數據: 4、寫數據: 5、常用的埠設置和方法: 串口最基本的功能就是實現通信,簡單來說就是讀和寫,就像大家熟 ...
  • Hi、大家好,今天又是美好的一天。 關於 Settings Sync 擴展: Settings Sync可以同步你當前的VSCode配置環境,當你需要在其它的電腦工作時,您不用重頭再來一遍。新機器登錄一下就搞定了。再也不用折騰環境了。 大致原理:使用GitHub Gist來同步多台電腦上的設置,代 ...
  • ASP.NET MVC 分頁使用的是作者楊濤的MvcPager分頁控制項 地址:http://www.webdiyer.com/mvcpager/demos/ajaxpaging/ 這個分頁控制項在裡面有很好的的案例,及註意事項 分頁在我們的項目中是經常需要使用到的,普通分頁體驗是在是太差了,每一次點擊 ...
  • 分享基於EF6、Unitwork、Autofac的Repository模式設計 [TOC] 一、實現的思路和結構圖 Repository的共同性 有一些公共的方法(增刪改查), 這些方法無關於Repository操作的是哪個實體類,可以把這些方法定義成介面IRepository,然後有個基類Base ...
  • 鏈接:https://pan.baidu.com/s/1lbTL8UNQr4o45Z30J_YGLA提取碼:xr3z 複製這段內容後打開百度網盤手機App,操作更方便哦 ...
  • 先看一下系統自帶的線型文件acadiso.lin: 因為STANDARD是每個CAD文檔必須要有的文字樣式,同樣的,如果想更改系統自定義的帶文字的線型樣式,需要更改STANDARD 需要註意的是,文字(管線)兩側的空白並不對稱,這是因為文字(管線)或圖形的長度實際是占用了下一個段落的長度, 這裡就是 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...