普通函数

string Func()
{
string x = X();
string y = Y();
string z = Z(); return x + y + z;
}

X(), Y(), Z()内部都Sleep 10s, 则Func()耗时30s

异步函数

async string FuncAsync()
{
string x = await XAsync();
string y = await YAsync();
string z = await ZAsync(); return x + y + z;
}

假设 XAsync(), YAsync(), YAsync()内部都Sleep10s, 则FuncAync()还是耗时多少呢?

整个FuncAsync会被拆成如下部分:

  1. FuncAsync-开始部分();
  2. XAsync();
  3. FuncAsync-中间部分1();
  4. YAsync();
  5. FuncAsync-中间部分2();
  6. ZAsync();
  7. FuncAsync-结束部分();

这些部分会被.NET同步上下文SynchronizationContext在同一个线程依次执行,如下:

  1. 同步上下文执行 FuncAsync-开始部分();
  2. 同步上下文执行 XAsync(); //内部调用了Sleep10s, 要10s才返回
  3. 同步上下文执行 FuncAsync-中间部分1();
  4. 同步上下文执行 YAsync();//内部调用了Sleep10s, 要10s才返回
  5. 同步上下文执行 FuncAsync-中间部分2()
  6. 同步上下文执行 ZAsync();//内部调用了Sleep10s, 要10s才返回
  7. 同步上下文执行 FuncAsync-结束部分();

所以耗时时间还是30s

假设 XAsync(), YAsync(), YAsync()内部都把Sleep10s用Task.Run()都包起来返回

  1. 同步上下文执行 FuncAsync-开始部分();
  2. 同步上下文执行 XAsync(); 立刻返回, 在另一个线程执行Sleep(10)
  3. 同步上下文等待 但可以执行其他异步任务, 如果是Winform的话UI不会卡
  4. 后台线程10s结束告诉同步上下文继续执行
  5. 同步上下文执行 FuncAsync-中间部分1();
  6. 同步上下文执行 YAsync(); 立刻返回, 在另一个线程执行Sleep(10)
  7. 同步上下文等待 但可以执行其他异步任务, 如果是Winform的话UI不会卡
  8. 后台线程10s结束告诉同步上下文继续执行
  9. 同步上下文执行 FuncAsync-中间部分2()
  10. 同步上下文执行 ZAsync(); 立刻返回, 在另一个线程执行Sleep(10)
  11. 同步上下文等待 但可以执行其他异步任务, 如果是Winform的话UI不会卡
  12. 后台线程10s结束告诉同步上下文继续执行
  13. 同步上下文执行 FuncAsync-结束部分();

所以FuncAsync总耗时时间还是30s,优点是UI线程不会被卡住, UI线程不是被FuncAsync一个任务占有,其他异步任务也能在UI线程上执行,真正等待的任务都运行在后台线程。

下面这个版本,FuncAsync()执行的时间才是10s

async string FuncAsync()
{
var taskX = XAsync();
var taskY = YAsync();
var taskZ = ZAsync(); await Task.WhenAll(taskX, taskY, taskZ); return taskX.Result + taskY.Result + taskZ.Result;
}

如果在XAsync()里调用Task.Result同步等待会怎样?

  1. 同步上下文执行 FuncAsync-开始部分();
  2. 同步上下文执行 XAsync();    
    1.   XAsync() 调用 YAsync()  返回task             
      1.   YAsync() 开启后台线程sleep10s
    2.   XAsync()等待task.Result    
  3. 后台线程10s结束告诉同步上下文继续

XAsync要等待的结果(task.Result),需要同步上下文返回,

虽然后台线程已经同步上下文结果了,但同步上下文此时却还在执行XAsync的过程中.

互相等待,于是产生死锁.

举个形象的例子:

  1. 医生Doctor给病人Patients看病, 病人必须先排队,按次序来。
  2. 轮到某个病人PatientX时要某项检查报告,病人说,等我一下,检查报告在我老婆手里,我打电话让她送来。
  3. 于是医生Doctor和病人PatientX开始等待, 左等右等不来。
  4. 其实几分钟后病人的老婆PatientXWife已经把检查报告送来了, 只不过她拿着检查报告在队列里等。

async/await的一些用法的更多相关文章

  1. js中回调函数,promise 以及 async/await 的对比用法 对比!!!

    在编程项目中,我们常需要用到回调的做法来实现部分功能,那么在js中我们有哪些方法来实现回调的? 方法1:回调函数 首先要定义这个函数,然后才能利用回调函数来调用! login: function (f ...

  2. [.NET] 怎样使用 async & await 一步步将同步代码转换为异步编程

    怎样使用 async & await 一步步将同步代码转换为异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6079707.html  ...

  3. [.NET] 利用 async & await 进行异步 IO 操作

    利用 async & await 进行异步 IO 操作 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6082673.html  序 上次,博主 ...

  4. async/await,了解一下?

    上一篇博客我们在现实使用和面试角度讲解了Promise(原文可参考<面向面试题和实际使用谈promise>),但是Promise 的方式虽然解决了 callback hell,但是这种方式 ...

  5. promise async await使用

    1.Promise (名字含义:promise为承诺,表示其他手段无法改变) Promise 对象代表一个异步操作,其不受外界影响,有三种状态: Pending(进行中.未完成的) Resolved( ...

  6. promise 进阶 —— async / await 结合 bluebird

    一.背景 1.Node.js 异步控制 在之前写的 callback vs async.js vs promise vs async / await 里,我介绍了 ES6 的 promise 和 ES ...

  7. javascript异步编程 Async/await

    Async/await Async/await 在学习他之前应当补充一定的 promise 知识 它是一种与 promise 相配合的特殊语法,目前被认为是异步编程的终级解决方案 值得我们每一个人学习 ...

  8. async,await,Task 的一些用法

    async,await,Task 的一些用法 private void Form1_Load(object sender, EventArgs e) { Display(); } public asy ...

  9. 异步操作之 Promise 和 Async await 用法进阶

    ES6 提供的 Promise 方法和 ES7 提供的 Async/Await 语法糖都可以更好解决多层回调问题, 详细用法可参考:https://www.cnblogs.com/cckui/p/99 ...

随机推荐

  1. 2017北京赛区J题

    类型:三维动态规划 题目链接 题意: 合并连续石头块,最终要合并成一块,求时间最短,每次只能连续合并L~R块石头,不能合并成一块时输出-1 题解: 利用动态规划解决两种分问题 dp[l][r][k]: ...

  2. Linux输入输出管理

      一.系统输入输出的理解 运行一个程序时,需要从某个位置读取输入信息,然后CPU处理,最后将输出 显示在屏幕或文件中:其中,某个位置相当于输入设备,屏幕或文件为输出设备. 标准输入:stdin,默认 ...

  3. Memcached 补充

    Memcached 补充 Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站 ...

  4. Git 之 问题集锦

    准备:远程仓库名:origin     远程分支:master.tt      本地分支:master.test 1. error: src refspec *** does not match an ...

  5. (转)MapReduce Design Patterns(chapter 6 (part 2))(十二)

    Chain Folding 这是对job 链的一种优化.基本上是一种大体规则:每条记录都会提交给多个mapper,或者给reducer然后给mapper.这种综合处理方法会节省很多读文件和传输数据的时 ...

  6. SpringMVC札集(04)——SpringMVC传递参数

    自定义View系列教程00–推翻自己和过往,重学自定义View 自定义View系列教程01–常用工具介绍 自定义View系列教程02–onMeasure源码详尽分析 自定义View系列教程03–onL ...

  7. ubuntu版的sublime-text3输入中文问题

    前言 sublime-text3中看又中用,但是呢,ubuntu下有个问题,就是无法输入中文.这很难受啊,要写一些重要的内容时,总还是中文来的顺畅. 解决办法 工具:sublime-text-imfi ...

  8. js之数码时钟加随机变色

      HTML <!DOCTYPE html> <html lang="en"> <head> <meta charset="UT ...

  9. excel中多表汇总

    excel中将多个表中的结果汇总到一张表格里,而且汇总表格的数据是随着其它表格的数据变化面变化 其实就是函数sumif的使用,sumif函数的结构: sumif函数语法 sumif(range,cri ...

  10. BJOI 2019 模拟赛 #2 题解

    T1 完美塔防 有一些空地,一些障碍,一些炮台,一些反射镜 障碍会挡住炮台的炮, 反射镜可以 90° 反射炮台的光线,炮台可以选择打他所在的水平一条线或者竖直一条线 求是否有一组方案满足每个空地必须要 ...