Async Programming - 1 async-await 糖的本质(1)
这一个系列的文章主要来讲 C# 中的语言特性 async-await 在语言层面的本质,我们都知道 await 是编译器进行了一个 rewrite,然而这个 rewrite 并不是直接 rewrite 成其他没有原生支持 await 的语言的 lambda 回调的形式,而是整个对方法进行了重写,下面就让我们来从最简单的方法,一步一步剖析 await 糖的工作机制。
一个 async 方法,就是你在代码执行到一半的时候,告诉电脑:我要把函数返回,你先去干别的事情(比如 UI 操作),等我这边的事完成之后,再回复现场继续从刚才返回的地方执行。方法的执行是依靠状态机驱动的,一系列的 MoveNext 方法推动状态机的执行,编译器则会将方法分而治之,把 async 方法体拆分成许多部分,每一部分是一个状态机中的状态,放进 MoveNext 中
新建一个控制台工程,我们从最简单的 async 方法开始:FooAsync
编译器已经提示我这个 async 标识符没卵用了,没关系
然后我们再写一个方法来对照被编译器重写过的方法:FooAsync2
这还没完,一个异步方法中执行的各项操作,运算,中间暂停然后返回,最后 return 结果,是由状态机推动的,所以我们手动 rewrite 的 FooAsync2 方法中需要一个状态机,并且要实现系统的 IAsyncStateMachine 接口。
state machine 是一个 struct,默认情况下一个 async 方法不需要等待,所以我们不希望有一个在堆上的东西来增加我们的运行负担。
你可以看到这个接口要求两个方法,第二个我们之后再讲,你可能会奇怪为什么一个 IAsyncStateMachine 实例需要 SetStateMachine 另一个实例(后面会说,这其实是他自己),由于我们的 state machine 是一个 struct,所以当方法 await 的时候,整个栈就回退给其他方法来用了,所以你需要把这个 state machine 以及其他的参数转移到堆上,然后用这个方法来获得他自己,这个时候堆上的对象的运行时消耗才是值得的。而 MoveNext 就是之前我们说过的用来推动整个方法运行的方法。
方法里已经装了一个 state machine 的实例了,然后我们来看看谁要来驱动着方法一步一步向前走,当控制流回到方法内部的时候他需要来执行一些操作,尽管我们这里的方法什么都没干,我们还是要把所有的机构都弄好。我们需要一个 AsyncMethodBuilder,这个东西也已经在 System.Runtime.CompilerServices 里面提供了。
这个 method builder 放在了状态机内,因为方法可能返回而我们一直需要它,所以要借助 state machine 转移到栈上的同时把它一起转移了。AsyncVoidMethodBuilder 也是一个 struct,同样也是我们不希望堆上的东西增加额外的运行时负担。
method builder 是我们在控制流开始,await,返回的时候应该去使用的东西,这个东西应该在 MoveNext 函数中被使用,这里我们的方法什么都没干,所以 MoveNext 中只有一个状态的转移:开始->返回。返回通过 method builder 的 SetResult 方法完成,所有的 return 都会被 rewrite 成 SetResult。
最后方法怎么开始呢?我们看看 method builder 有没有 Start 方法,哈!有。Start 方法需要 state machine 的引用,为什么需要,因为 method builder 需要在方法一步步进行的时候调用 ModeNext,为什么要引用,这也说的通,方法的状态只需要一份,使用引用,避免拷贝,以及 state machine 有可能在栈上(现在这样)也有可能在堆上,需要用引用来指向它。
现在的 FooAsync2 就是 async 方法 FooAsync 被 rewrite 之后的样子。我们在 Main 里调用 FooAsync2,单步执行
从调用堆栈看,Start 方法调用了 MoveNext,然后我们的 MoveNext 方法就像 FooAsync 一样,什么都没干,直接 SetResult 然后返回,然后 Start 返回,FooAsync2 返回,Main 返回,一个没卵用的异步方法完成了。
这样就是一个最简单的异步方法被 rewrite 之后的样子,这一篇就到这里,下一篇讲讲稍微复杂点的方法。
Async Programming - 1 async-await 糖的本质(1)的更多相关文章
- Async Programming - 1 async-await 糖的本质(2)
上一篇讲了这么多,其实说的就是一个事,return会被编译器重写成SetResult,所以如果我们的异步函数返回的是一个Task<int>,代码就要改成这样: using System; ...
- C#的多线程——使用async和await来完成异步编程(Asynchronous Programming with async and await)
https://msdn.microsoft.com/zh-cn/library/mt674882.aspx 侵删 更新于:2015年6月20日 欲获得最新的Visual Studio 2017 RC ...
- Asynchronous programming with async and await (C#)
Asynchronous Programming with async and await (C#) | Microsoft Docs https://docs.microsoft.com/en-us ...
- Async Programming All in One
Async Programming All in One Async & Await Frontend (async () => { const url = "https:// ...
- [转] Scala Async 库 (Scala future, await, async)
[From] https://colobu.com/2016/02/15/Scala-Async/ 在我以前的文章中,我介绍了Scala Future and Promise.Future代表一个异步 ...
- async await 的 实质 本质
async await 的 实质 就是 用 “状态机” 来 取代 函数层层调用 . async await 的 本质 是 语法糖, 和 提高性能 什么的 没什么关系 . 为了避免理解歧义, 我把 ...
- Javascript 使用 async 声明符和 await 操作符进行异步操作
async function 声明用于定义一个返回 AsyncFunction 对象的异步函数 await 操作符用于等待一个Promise 对象.它只能在异步函数 async function 中 ...
- a kind of async programming in c#, need to reference definition
void Main() { Run d=new Run(RunHandler); IAsyncResult result= d.BeginInvoke(new AsyncCallback(CallBa ...
- [Javascript] Await a JavaScript Promise in an async Function with the await Operator
The await operator is used to wait for a promise to settle. It pauses the execution of an async func ...
随机推荐
- winform在设置控件enabled=false后,无法更改控件字体颜色的问题
项目界面设计的时候,发现在设置button的enabled=false后,原本设计的字体颜色跟预设的不一样,查了一些资料后,在网上看到这样一段代码: [System.Runtime.InteropSe ...
- 如何全面解决ECSHOP的jquery冲突
主要就是Ecshop的AJAX传输类,transport.js中重写了object的对象原型,从而导致了与jq框架的冲突. 解决: 1. 删除transport.js中587行 - 636行中关于ob ...
- oracle数据库卸载(需要完全卸载oracl才能重装)
1. 开始->设置->控制面板->管理工具->服务 停止所有Oracle服务. 2. 开始->程序->Oracle - OraHome81->Oracle I ...
- Model--汇总
NSFileManager.NSURL.NSFileHandle.NSData.NSXMLParser.NSUserDefaults.NSKeyedArchiver.NSKeyedUnarchiver ...
- discuz不能发表投票、附件上传等
如何开启和发表投票主题 本教程主要讲述如何开启和发表投票的帖子.具体的操作如下: 1.论坛后台 => 用户 => 用户组 => 详情 => 特殊主题 => 允许发起投票设 ...
- 关于MOOC的学习总结
20125133 马国祥 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 对 ...
- Object.observe
使用Object.observe 实现数据绑定
- ng-class的用法
最近在学习angular框架,ng-class是angular框架的一个指令,这里是ng-class指令的官方解释: ng-class 指令用于给 HTML 元素动态绑定一个或多个 CSS 类. ng ...
- About Flash
Take SAMSUNG K9F1G08U0E for example, and use S3C2416, Windows CE5.0 platform. 要让Flash工作起来,包括两部分: Fla ...
- 字符串hash算法
http://www.cnblogs.com/zyf0163/p/4806951.html hash函数对大家来说不陌生吧 ? 而这次我们就用hash函数来实现字符串匹配. 首先我们会想一下二进制数. ...