深入理解Async/Await(转)

C# 5 Async/Await 语法特性,极大地简化了异步编程,但我们知道,异步编程的基本原理并没有发生根本改变。也就是说,当一些复杂的东西看起来很简单时,它通常意味着有一些有趣的事情在背后发生。在计算机程序设计语言领域,我们把这些本身很复杂但看起来很简单的语言特性称为语法糖,通常情况下,我们并不需要深入理解语法糖是怎么被一层一层包裹起来的,但是,最近我在使用.NET Core实现MySQL协议过程中,需要实现一个Awaitable Socket,所以我需要知道Async/Await背后到底发上了什么?

编译器重写

我们通过写一个非常简单的控制台应用程序,一层一层地剥开C#编译器实现的 Async/Await 语法糖。

namespace AsyncAwaitInDepth{
    class Program
    {
        static void Main(string[] args){ }

        static async int Method()
        {
            return 1;
        }
    }}

编译上面的C#代码,会出现CS1983 The return type of an async method must be void, Task or Task<T>错误,即编译器告诉我们,异步方法仅限于三个不同的返回类型︰

  • void
  • Task
  • Task

再修改一下代码,让编译器编译通过:

}}

使用ILSpy或者Reflector看看编译器干了什么:

上面的代码可以发现,aynce/await 代码被C#编译器重写了,编译器通过使用三个Builder和动态生成的StateMachine替换掉了我们的代码,而把我们的代码整理到StateMachine的MoveNext()方法内部,三个Builder是:

  • AsyncVoidMethodBuilder
  • AsyncTaskMethodBuilder
  • AsyncTaskMethodBuilder

撇开这三个Builder的内部实现,从接口角度看,他们长得很像,我们只看看AsyncTaskMethodBuilder<T>接口:l

C#编译器动态生成的StateMachine实现了IAsyncStateMachine接口:

  • AsyncVoidMethodBuilder
  • AsyncTaskMethodBuilder
  • AsyncTaskMethodBuilder

我们自己代码中的await Task.Delay(5);表达式,在StateMachine的MoveNext()方法内部被替换成awaiter = Task.Delay(5).GetAwaiter();,然后通过IsCompleted属性判断是否需要启动一个新的线程去执行,最后通过awaiter.GetResult();获取结果,或者是给Builder设置异常builder.SetException(exception);

Awaitables and Awaiters

在.NET中,我们知道Xtable和Xter是一个一对的概念,比如IEnumerable<T>IEnumerator<T>,上面的代码,我们发现Task,Task<TResult>是可以被awaited的,他们是awaitable,可以通过其GetAwaiter()方法得到awaiter:

awaitableGetAwaiter()可以是实例方法,也可以是扩展方法。该方法返回的是一个awaiter,该awaiter必须实现INotifyCompletion接口,可选实现ICriticalNotifyCompletion接口,同时,必须提供一个名为IsCompleted bool 属性, 必须提供public TResult GetResult()方法。

自定义awaitable和awaiter对象

我们费了那么大的劲,终于弄清楚了 Async/Await 背后到底发上了什么,做到了知其然知其所以然,因此,我们可以实现自己的awaitable和awaiter对象:

最后,借用《C# in Depth》作者Job Skeet的一句话与大家共勉:

I like knowing how a feature works before I go too far using it

原文地址:http://www.xyting.org/2017/02/28/understand-async-await-in-depth.html

时间: 2024-10-21 01:46:39

深入理解Async/Await(转)的相关文章

理解 async/await

ES7 提出的async 函数,终于让 JavaScript 对于异步操作有了终极解决方案.No more callback hell.async 函数是 Generator 函数的语法糖.使用 关键字 async 来表示,在函数内部使用 await 来表示异步.想较于 Generator,Async 函数的改进在于下面四点: 内置执行器.Generator 函数的执行必须依靠执行器,而 Aysnc 函数自带执行器,调用方式跟普通函数的调用一样 更好的语义.async 和 await 相较于 *

理解ES7中的async/await

优势是:就是解决多层异步回调的嵌套 从字面上理解 async/await, async是 "异步"的含义,await可以认为是 async wait的简写,因此可以理解 async 用于声明一个function是异步的,而await用于等待一个异步方法执行完成返回的值(返回值可以是一个Promise对象或普通返回的值).注意:await 只能出现在 async函数中. 1-1 async的作用?首先来理解async函数是怎么处理返回值的,我们以前写代码都是通过return语句返回我们想

理解 JavaScript 的 async/await

随着 Node 7 的发布,越来越多的人开始研究据说是异步编程终级解决方案的 async/await.我第一次看到这组关键字并不是在 JavaScript 语言里,而是在 c# 5.0 的语法中.C# 的 async/await 需要在 .NET Framework 4.5 以上的版本中使用,因此我还很悲伤了一阵--为了要兼容 XP 系统,我们开发的软件不能使用高于 4.0 版本的 .NET Framework. 我之前在<闲谈异步调用"扁平"化> 中就谈到了这个问题.无论

进阶篇:以IL为剑,直指async/await

接上篇:30分钟?不需要,轻松读懂IL,这篇主要从IL入手来理解async/await的工作原理. 先简单介绍下async/await,这是.net 4.5引入的语法糖,配合Task使用可以非常优雅的写异步操作代码,它本身并不会去创建一个新线程,线程的工作还是由Task来做,async/await只是让开发人员以直观的方式写异步操作代码,而不像以前那样到处都是callback或事件. async/await IL翻译 先写个简单的例子: 1 using System; 2 using Syste

浅谈Async/Await

概要 在很长一段时间里面,FE们不得不依靠回调来处理异步代码.使用回调的结果是,代码变得很纠结,不便于理解与维护,值得庆幸的是Promise带来了.then(),让代码变得井然有序,便于管理.于是我们大量使用,代替了原来的回调方式.但是不存在一种方法可以让当前的执行流程阻塞直到promise完成.JS里面,我们无法直接原地等promise完成,唯一可以用于提前计划promise完成后的执行逻辑的方式就是通过then附加回调函数. 现在随着Async/Await的增加,可以让接口按顺序异步获取数据

Koa2学习(二)async/await

Koa2学习(二)async/await koa2中用到了大量的async/await语法,要学习koa2框架,首先要好好理解async/await语法. async/await顾名思义是一个异步等待的语法,是es7中为了实现用同步的方式写异步方法的一种新式语法. async 我们再来看看async到底是一个什么语法: 普通的方法: function syncF() { return 'I am a sync result' } let sync_result = syncF() console

从async await 报错Unexpected identifier 谈谈对上下文的理解

原文首发地址:http://www.cnblogs.com/lonhon/p/7518231.html 先简单介绍下async await: async/await是ES6推出的异步处理方案,目的也很明确:更好的实现异步编程.   详细见阮大神 ES6入门 现在说说实践中遇到的问题:使用await报错Unexpected identifier 先上代码: var sleep = function (time) { return new Promise(function (resolve, rej

async/await的实质理解

async/await关键字能帮助开发者更容易地编写异步代码.但不少开发者对于这两个关键字的使用比较困惑,不知道该怎么使用.本文就async/await的实质作简单描述,以便大家能更清楚理解. 一.async的实质 async的实质是告诉调用者,async标记的方法可能包含异步代码.具体来说,以wait以分界点,async标记的方法被分为三个片段: static async void TestAsyncMethod() { someCode1(); //片段1 await Thask.Run(.

理解koa2 之 async + await + promise

koa是下一代的Node.js web框架. 我们首先使用koa来实现一个简单的hello world吧!假如目前的项目结构如下: ### 目录结构如下: koa-demo1 # 工程名 | |--- app.js # node 入口文件 | |--- node_modules # 项目依赖包 | |--- package.json app.js 代码如下: const Koa = require('koa'); const app = new Koa(); app.use(async (ctx