async 和 await 之非同步編程的學習

来源:https://www.cnblogs.com/cqhaibin/archive/2018/01/07/8232418.html
-Advertisement-
Play Games

async修改一個方法,表示其為非同步方法。而await表示等待一個非同步任務的執行。js方面,在es7中開始得以支持;而.net在c#5.0開始支持。本文章將分別簡單介紹他們在js和.net中的基本用法。 一、在js中的實現 js中的非同步,還是基於Promise實現的。沒有Promise就辦法談非同步了 ...


      async修改一個方法,表示其為非同步方法。而await表示等待一個非同步任務的執行。js方面,在es7中開始得以支持;而.net在c#5.0開始支持。本文章將分別簡單介紹他們在js和.net中的基本用法。

一、在js中的實現

js中的非同步,還是基於Promise實現的。沒有Promise就辦法談非同步了。並且await只能出現async修改的方法中;以及reject會觸發catch(異常)。

class AsyncTest{
    //simple example
    async run(){
        //按照順序等待後輸出
        let one = await this.output("one", 1000);
        console.log('output:' + one);
        let two = await this.output("two", 3000);
        console.log(two);
        console.log('run.....');
    }
    //await and Promise.all difference
    async runDiff(){ 
        let one = this.output('diff one', 2000);
        let two = this.output('diff two', 2000);
        console.log(  await two + await one );  //在2秒之後,兩個都輸出了,而不是各自都等待兩秒
        console.log('runDiff.....');
    }

    //Promise.all realize
    runAll(){
        let nowTime = new Date();
        console.log('b:' + nowTime.toTimeString());
        let array = ["a", "b", "c"];
        let that = this;
        array.forEach(async function(item){
            console.log( await that.output(item, 2000) );//2秒後同時輸出
        });
        let fn = async ()=>{
            for(let item of array){
                let v = await this.output(item, 2000);
                console.log(v ); //分步驟兩秒執行
            }
        }
        fn.call(this);
    }

    premosFn(){
        let nowTime = new Date();
        console.log('b:' + nowTime.toTimeString());
        let array = ["a", "b", "c"];
        let that = this;
        //promise.all
        let preFn = async function(){
            let promises = array.map(function(item){
                return that.output(item,2000); //同時開啟多個定時器
            });
            let r = await Promise.all(promises);
            console.log(r.join(','));
        }
        preFn();
    }

    reject(){
        let rejectFn = function(){
            return new Promise((resolve, reject)=>{
                setTimeout(()=>{
                    reject();
                },2000);
            });
        }
        let asyncReject = async function(){
            try{
                await rejectFn();
            }catch( e) {
                console.log('reject.....');
            }
        }
        asyncReject();
    }

    output(log, time){
        return new Promise(resolve=>{
            setTimeout(()=>{
                var nowTime = new Date();
                resolve( nowTime.toTimeString() + ":" + log + "\r\n");
            }, time);
        });
    }
}

方法說明如下:

  • output:簡單的輸出方法,但返回了一Promise。
  • run: 使用await來等待兩次對output的執行
  • runDiff:調用output時即創建promise。兩個promise會同步執行
  • runAll:多任務同步執行和按步驟執行的實現方法。也就是forEach和for方法體中使用await的區別
  • premosFn: promise.all的使用。
  • reject: promise的reject會觸發await的異常。

二、在c#中的實現

C#中異常是通過Task來實現的,所以標記了async的方法,其方法體中都可以出現await,否則不可以。以及Task中拋出的異常,如果沒有同步等待,則不能獲取異常

public class AsyncDemo
{
    private Task<string> Output(string val, int time)
    {
        return System.Threading.Tasks.Task.Run(() =>
        {
            System.Threading.Thread.Sleep(time * 1000);
            return (DateTime.Now.ToLongTimeString()) + ": " + val + "\r\n";
        });
    }

    public async System.Threading.Tasks.Task Run()
    {
        string oneVal = await Output("One", 2);
        string twoVal = await Output("Two", 2);
        System.Console.WriteLine("Run \r\n" + oneVal + " " + twoVal);
    }

    public async System.Threading.Tasks.Task RunDiff()
    {
        Task<string> oneTask = Output("one", 2);
        Task<string> twoTask = Output("two", 2);
        string val = await oneTask + await twoTask; 
        System.Console.WriteLine("RunDiff \r\n" + val);

    }

    public async System.Threading.Tasks.Task RunAll()
    {
        System.Console.WriteLine("b:" + (DateTime.Now.ToLongTimeString()));

        string[] array = new string[3] { "a", "b", "c" };
        foreach(var item in array)
        {
            string v = await Output(item, 2);
            System.Console.WriteLine(v);
        }
    }

    public async System.Threading.Tasks.Task PromiseFn()
    {
        System.Console.WriteLine("b:" + (DateTime.Now.ToLongTimeString()));

        string[] array = new string[3] { "a", "b", "c" };
        List<System.Threading.Tasks.Task<string>> tasks = new List<System.Threading.Tasks.Task<string>>();
        foreach (var item in array)
        {
            tasks.Add(Output(item, 2));
        }
        //waitAll返回值不能獲取,他返回為void,而WhenAll則返回為一個Task(這個Task就有其列表值)
        string[] r = await System.Threading.Tasks.Task.WhenAll(tasks.ToArray());
        System.Console.WriteLine(string.Join(",",r));
    }

    public async System.Threading.Tasks.Task Reject()
    {
        Func<System.Threading.Tasks.Task> func = async () =>
        {
            throw new Exception("custom...");
            await Output("reject", 2);
        };
        await func();
    }
}

調用代碼如下:

AsyncDemo asyncDemo = new AsyncDemo();
asyncDemo.Run().Wait();
asyncDemo.RunDiff().Wait();
asyncDemo.RunAll().Wait();
asyncDemo.PromiseFn().Wait();
try
{
    asyncDemo.Reject().Wait();
}catch(Exception e)
{
    System.Console.WriteLine("reject ex");
}

    上述代碼就是Js的async和await在c#中的翻版實現。 其中每個非同步方法的調用,都用到了Wait方法來進行同步等待。以獲取到結果。而沒有像Js中那麼難以控制。尤其註意,async方法中異常的捕獲。

三、兩者的異同點

  • js中的async方法的調用,是沒有wait方法來等待結果的執行的,只能通過promise來監聽執行結果
  • c#中的async方法,由於推薦返回Task或者Task<T>,所以可以用Wait來等待執行結果,如果async方法返回為void,則與js類似。 C#中的下麵示例方法的調用者捕獲不了異常:
    public async void Run()
    {
        string oneVal = await Output("One", 2);
        string twoVal = await Output("Two", 2);
        System.Console.WriteLine("Run" + oneVal + " " + twoVal);
    }
  • c#中的Task可以非同步方法的鏈式調用,即可將前一任務的執行結果作為第二任務的參數傳入,當然js的Promise也完全是可以很輕鬆的實現:
System.Console.WriteLine("b:" + (DateTime.Now.ToLongTimeString()));

string[] array = new string[3] { "a", "b", "c" };

var r = await this.Output(array.First(), 2).ContinueWith((tsc) =>
{
    string v = tsc.GetAwaiter().GetResult();
    return this.Output(v + "02", 2);
});
System.Console.WriteLine(r.Result);

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

-Advertisement-
Play Games
更多相關文章
  • 介面: 暫時可以理解為是一種特殊的抽象類 介面是功能的集合,可以看作是一種數據類型,是比抽象類更抽象的“類” 介面只描述所應該具備的方法,並沒有具體實現,具體實現由介面的實現類(相當於介面的子類)來完成 這樣將功能的實現與定義分離,優化了程式設計 介面的成員方法全抽象,不能存在帶有方法體的方法 介面 ...
  • 1,關於Spring MVC的核心控制器DispatcherServlet的作用,以下說法錯誤的是( ) A,它負責接收HTTP請求 B,載入配置文件 C,實現業務操作 D,初始化上下應用對象ApplicationContext 正確答案:C SpringMVC是Spring中的模塊,它實現了mvc ...
  • 準備工作: 在web.xml中配置shiro核心過濾器 在spring配置文件中提供核心過濾器運行所需要的輔助bean對象,在對象內註入安全管理器 攔截認證 配置三個url 攔截除了登錄頁面以及認證action之外所有請求 編寫shiro認證 編寫自己login的action 編寫自定義realm ...
  • http://blog.csdn.net/chgaowei/article/details/6427731 為了支持c++的多態性,才用了動態綁定和靜態綁定。理解他們的區別有助於更好的理解多態性,以及在編程的過程中避免犯錯誤。 需要理解四個名詞: 1、對象的靜態類型:對象在聲明時採用的類型。是在編譯 ...
  • Java提供了大量持有對象的方式: (1) 數組將數字與對象聯繫起來。 它保存類型明確的對象,查詢對象時,不需要對結果做類型轉換。它可以是多維的, 可以保存基本類型的數據。 但是,數組一旦生成,其容量就不能改變。 (2)Collection保存單一的元素,而Map保存相關聯的鍵值對。有了Java的泛 ...
  • 1.hashCode()和equals() API hashCode()和equals()都來自上帝類Object, 所有的類都會擁有這兩個方法,特定時,覆寫它們。 它們是用來在同一類中做比較用的,尤其是在容器里如Set存放同一類對象時用來判斷放入的對象是否重覆。 下麵是API中的介紹: boole ...
  • 問題原因: 1.iis是在安裝完.net framework 之後才安裝的,需要進行iis註冊,開始--運行--cmd,打開命令行提示符,輸入命令如下 C:\Windows\Microsoft.NET\Framework\v4.0.30319\aspnet_regiis.exe -i 2.如果伺服器 ...
  • 背水一戰 Windows 10 之 用戶和賬號: 微軟賬號的登錄和註銷 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...