本文主要是介绍async 和 await 实现原理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
来源:http://blog.csdn.net/cjq1234/article/details/7536644
摘要:用串行运行的状态机模拟实现了异步
今天在stackOverflow网站看到一个很好的解释,摘抄并发挥一下,
It works similarly to the yield return keyword in C# 2.0.
An asynchronous method is not actually an ordinary sequential method. It is compiled into a state machine (an object) with some state (local variables are turned into fields of the object). Each block of code between two uses of await is one "step" of the state machine.
This means that when the method starts, it just runs the first step and then the state machine returns and schedules some work to be done - when the work is done, it will run the next step of the state machine. For example this code:
- async Task Demo() {
- var v1 = foo();
- var v2 = await bar();
- more(v1, v2);
- }
async Task Demo() {var v1 = foo();var v2 = await bar();more(v1, v2);
}
Would be translated to something like:
- class _Demo {
- int _v1, _v2;
- int _state = 0;
- Task<int> _await1;
- public void Step() {
- switch(this._state) {
- case 0:
- foo();
- this._v1 = foo();
- this._await1 = bar();
- // When the async operation completes, it will call this method
- this._state = 1;
- op.SetContinuation(Step);
- case 1:
- this._v2 = this._await1.Result; // Get the result of the operation
- more(this._v1, this._v2);
- }
- }
class _Demo {int _v1, _v2;int _state = 0;Task<int> _await1;public void Step() {switch(this._state) {case 0:foo();this._v1 = foo();this._await1 = bar();// When the async operation completes, it will call this methodthis._state = 1;op.SetContinuation(Step);case 1:this._v2 = this._await1.Result; // Get the result of the operationmore(this._v1, this._v2);}
}
The important part is that it just uses the SetContinuation method to specify that when the operation completes, it should call the Step method again (and the method knows that it should run the second bit of the original code using the _state field). You can easily imagine that the SetContinuation would be something like btn.Click += Step, which would run completely on a single thread.
The asynchronous programming model in C# is very close to F# asynchronous workflows (in fact, it is essentially the same thing, aside from some technical details), and writing reactive single-threaded GUI applications using async is quite an interesting area - at least I think so - see for example this article (maybe I should write a C# version now :-)).
The translation is similar to iterators (and yield return) and in fact, it was possible to use iterators to implement asynchronous programming in C# earlier. I wrote an article about that a while ago - and I think it can still give you some insight on how the translation works.
answered by Tomas Petricek
大致是说 async 和await模拟了一个状态机(上面的_state变量维持状态),async把一个顺序执行的工作分割成3块,一个是费时工作的前序,一个是费时的工作,还有一个是费时工作的后续,
- async Task Demo() {
- var v1 = 前序工作();
- var v2 = await 费时工作();
- 后续工作(v1, v2);
- }
async Task Demo() {var v1 = 前序工作();var v2 = await 费时工作();后续工作(v1, v2);
}
被翻译成状态机
- class 状态机{
- int _v1, _v2;
- int 状态变量 =起始态;
- Task<int> _await1;
- public void Step() {
- switch(this.状态变量) {
- case 起始态:
- this._v1 = 前序工作();
- this._await1 = 费时工作();
- // 当费时工作异步完成时, 异步工作将改变状态变量值
- this.状态变量= 顺利完成;
- 继续调用自身即 Step函数; //op.SetContinuation(Step);
- case 顺利完成:
- this._v2 = this._await1.Result; //费时工作结果
- 后续工作(this._v1, this._v2);
- }
- }
class 状态机{int _v1, _v2;int 状态变量 =起始态; Task<int> _await1; public void Step() { switch(this.状态变量) { case 起始态: this._v1 = 前序工作();this._await1 = 费时工作(); // 当费时工作异步完成时, 异步工作将改变状态变量值this.状态变量= 顺利完成; 继续调用自身即 Step函数; //op.SetContinuation(Step); case 顺利完成: this._v2 = this._await1.Result; //费时工作结果 后续工作(this._v1, this._v2); }
}
事实上我们亦可以在.net framework 4以下向上面的状态机类一样来调用异步函数,这样同步调用的逻辑还是保持住了,也更好理解,最大的好处是对于循环的处理简单多了。
这篇关于async 和 await 实现原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!