关键:

异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。

async/await 结构可分成三部分:

(1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;

(2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;

(3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。

一、What's 异步?

启动程序时,系统会在内存中创建一个新的进程。进程是构成运行程序资源的集合。

在进程内部,有称为线程的内核对象,它代表的是真正的执行程序。系统会在 Main 方法的第一行语句就开始线程的执行。

线程:

①默认情况,一个进程只包含一个线程,从程序的开始到执行结束;

②线程可以派生自其它线程,所以一个进程可以包含不同状态的多个线程,来执行程序的不同部分;

③一个进程中的多个线程,将共享该进程的资源;

④系统为处理器执行所规划的单元是线程,而非进程。

一般来说我们写的控制台程序都只使用了一个线程,从第一条语句按顺序执行到最后一条。但在很多的情况下,这种简单的模型会在性能或用户体验上不好。

例如:服务器要同时处理来自多个客户端程序的请求,又要等待数据库和其它设备的响应,这将严重影响性能。程序不应该将时间浪费在响应上,而要在等待的同时执行其它任务!

现在我们开始进入异步编程。在异步程序中,代码不需要按照编写时的顺序执行。这时我们需要用到 C# 5.0 引入的 async/await 来构建异步方法。

我们先看一下不用异步的示例:

class Program{//创建计时器private static readonly Stopwatch Watch = new Stopwatch();private static void Main(string[] args){//启动计时器Watch.Start();const string url1 = "http://www.cnblogs.com/";const string url2 = "http://www.cnblogs.com/liqingwen/";//两次调用 CountCharacters 方法(下载某网站内容,并统计字符的个数)var result1 = CountCharacters(1, url1);var result2 = CountCharacters(2, url2);//三次调用 ExtraOperation 方法(主要是通过拼接字符串达到耗时操作)for (var i = 0; i < 3; i++){ExtraOperation(i + 1);}//控制台输出Console.WriteLine($"{url1} 的字符个数:{result1}");Console.WriteLine($"{url2} 的字符个数:{result2}");Console.Read();}/// <summary>/// 统计字符个数/// </summary>/// <param name="id"></param>/// <param name="address"></param>/// <returns></returns>private static int CountCharacters(int id, string address){var wc = new WebClient();Console.WriteLine($"开始调用 id = {id}:{Watch.ElapsedMilliseconds} ms");var result = wc.DownloadString(address);Console.WriteLine($"调用完成 id = {id}:{Watch.ElapsedMilliseconds} ms");return result.Length;}/// <summary>/// 额外操作/// </summary>/// <param name="id"></param>private static void ExtraOperation(int id){//这里是通过拼接字符串进行一些相对耗时的操作var s = "";for (var i = 0; i < 6000; i++){s += i;}Console.WriteLine($"id = {id} 的 ExtraOperation 方法完成:{Watch.ElapsedMilliseconds} ms");}}

图1-1 运行的效果图,以毫秒(ms)为单位

  【备注】一般来说,直接拼接字符串是一种比较耗性能的手段,如果对字符串拼接有性能要求的话应该使用 StringBuilder。

  【注意】每次运行的结果可能不同。不管哪次调试,绝大部分时间都浪费前两次调用(CountCharacters 方法),即在等待网站的响应上。

                                                              图1-2 根据执行结果所画的时间轴

有人曾幻想着这样提高性能的方法:在调用 A 方法时,不等它执行完,直接执行 B 方法,然后等 A 方法执行完成再处理。

C# 的 async/await 就可以允许我们这么弄。

class Program{//创建计时器private static readonly Stopwatch Watch = new Stopwatch();private static void Main(string[] args){//启动计时器Watch.Start();const string url1 = "http://www.cnblogs.com/";const string url2 = "http://www.cnblogs.com/liqingwen/";//两次调用 CountCharactersAsync 方法(异步下载某网站内容,并统计字符的个数)Task<int> t1 = CountCharactersAsync(1, url1);Task<int> t2 = CountCharactersAsync(2, url2);//三次调用 ExtraOperation 方法(主要是通过拼接字符串达到耗时操作)for (var i = 0; i < 3; i++){ExtraOperation(i + 1);}//控制台输出Console.WriteLine($"{url1} 的字符个数:{t1.Result}");Console.WriteLine($"{url2} 的字符个数:{t2.Result}");Console.Read();}/// <summary>/// 统计字符个数/// </summary>/// <param name="id"></param>/// <param name="address"></param>/// <returns></returns>private static async Task<int> CountCharactersAsync(int id, string address){var wc = new WebClient();Console.WriteLine($"开始调用 id = {id}:{Watch.ElapsedMilliseconds} ms");var result = await wc.DownloadStringTaskAsync(address);Console.WriteLine($"调用完成 id = {id}:{Watch.ElapsedMilliseconds} ms");return result.Length;}/// <summary>/// 额外操作/// </summary>/// <param name="id"></param>private static void ExtraOperation(int id){//这里是通过拼接字符串进行一些相对耗时的操作var s = "";for (var i = 0; i < 6000; i++){s += i;}Console.WriteLine($"id = {id} 的 ExtraOperation 方法完成:{Watch.ElapsedMilliseconds} ms");}}//这是修改后的代码

图1-3 修改后的执行结果图

图1-4 根据加入异步后的执行结果画的时间轴。

  我们观察时间轴发现,新版代码比旧版快了不少(由于网络波动的原因,很可能会出现耗时比之前长的情况)。这是由于 ExtraOperation 方法的数次调用是在 CountCharactersAsync 方法调用时等待响应的过程中进行的。所有的工作都是在主线程中完成的,没有创建新的线程。

  【改动分析】只改了几个细节的地方,直接展开代码的话可能看不出来,改动如下:

                              

图1-5

                                                                                          图1-6

  ①从 Main 方法执行到 CountCharactersAsync(1, url1) 方法时,该方法会立即返回,然后才会调用它内部的方法开始下载内容。该方法返回的是一个 Task<int> 类型的占位符对象,表示计划进行的工作。这个占位符最终会返回 int 类型的值。

  ②这样就可以不必等 CountCharactersAsync(1, url1) 方法执行完成就可以继续进行下一步操作。到执行 CountCharactersAsync(2, url2)  方法时,跟 ① 一样返回 Task<int> 对象。

  ③然后,Main 方法继续执行三次 ExtraOperation 方法,同时两次 CountCharactersAsync 方法依然在持续工作 。

  ④t1.Result 和 t2.Result 是指从 CountCharactersAsync 方法调用的 Task<int> 对象取结果,如果还没有结果的话,将阻塞,直有结果返回为止。

二、async/await 结构

先解析一下专业名词:

同步方法:一个程序调用某个方法,等到其执行完成之后才进行下一步操作。这也是默认的形式。

异步方法:一个程序调用某个方法,在处理完成之前就返回该方法。通过 async/await 我们就可以实现这种类型的方法。

async/await 结构可分成三部分:

(1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;

(2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;

(3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。

  现在我们来分析一下示例。

                                                                                            图2-1

三、What’s 异步方法

异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。

语法分析:

(1)关键字:方法头使用 async 修饰。

(2)要求:包含 N(N>0) 个 await 表达式(不存在 await 表达式的话 IDE 会发出警告),表示需要异步执行的任务。

(3)返回类型:只能返回 3 种类型(void、Task 和 Task<T>)。Task 和 Task<T> 标识返回的对象会在将来完成工作,表示调用方法和异步方法可以继续执行。

(4)参数:数量不限,但不能使用 out 和 ref 关键字。

(5)命名约定:方法后缀名应以 Async 结尾。

(6)其它:匿名方法和 Lambda 表达式也可以作为异步对象;async 是一个上下文关键字;关键字 async 必须在返回类型前。

图3-1 异步方法的简单结构图

 

async/await的优雅的打开方式是这样的:

private async void button1_Click(object sender, EventArgs e){var t = Task.Run(() => {Thread.Sleep(5000);return "Hello I am TimeConsumingMethod";});textBox1.Text = await t;}

寥寥几行就搞定了,不用再多写那么多函数,使用起来也很灵活。最让人头疼的跨线程修改控件的问题完美解决了,再也不用使用Invoke了,因为修改控件的操作压根就是在原来的线程上做的,还能不阻塞UI。

【转】C# 彻底搞懂async/await相关推荐

  1. C# 彻底搞懂async/await

    前言 Talk is cheap, Show you the code first! private void button1_Click(object sender, EventArgs e) {C ...

  2. 一次性搞懂JavaScript 执行机制

    你是否遭受到这样的恐吓? 你是否有过每个表达式前面都console一遍值去找执行顺序? 看了很多js执行机制的文章似乎都是似懂非懂,到技术面问的时候,理不清思绪.总结了众多文章的例子和精华,希望能帮到 ...

  3. 异步编程:一次搞懂Promise,async,await

    文章目录 前言 一.回调函数 二.Promise 三.错误处理 四.async/await await使用时的陷阱 1 2 3 总结 前言 异步编程允许我们在执行一个长时间任务时,程序不需要进行等待, ...

  4. 一文看懂async和“await”关键词是如何简化了C#中多线程的开发过程

    一文看懂"async"和"await"关键词是如何简化了C#中多线程的开发过程 当我们使用需要长时间运行的方法(即,用于读取大文件或从网络下载大量资源)时,在同 ...

  5. 一眼看懂promise与async await的区别

    // promise方法let p1 = new Promise((resolve,reject) => {setTimeout(() => {resolve('我是p1')},4000) ...

  6. 在.NET中执行Async/Await的两种错误方法

    微信公众号:架构师高级俱乐部 关注可了解更多的编程,架构知识.问题或建议,请公众号留言; 如果你觉得此文对你有帮助,欢迎转发 在.NET中执行异步/等待的两种错误方法 在应用开发中,我们为了提高应用程 ...

  7. 转:前端 100 问:能搞懂80%的请把简历给我

    <前端 100 问:能搞懂80%的请把简历给我> 引言 半年时间,几千人参与,精选大厂前端面试高频 100 题,这就是「壹题」. 在 2019 年 1 月 21 日这天,「壹题」项目正式开 ...

  8. 这次彻底搞懂 Promise(手写源码多注释篇)

    作者:一阵风,一枚只想安静写代码的程序员,来自程序员成长指北交流群    github: https://github.com/yizhengfeng-jj/promise 前言 promise 是 ...

  9. 【JS】1015- 异步编程的终极解决方案 async/await

    早期的回调函数 回调函数我们经常有写到,比如: ajax(url, (res) => {console.log(res); }) 复制代码 但是这种回调函数有一个大缺陷,就是会写出 回调地狱(C ...

最新文章

  1. 使用模块化编译缩小 apk 体积
  2. [AaronYang]C#人爱学不学8[事件和.net4.5的弱事件深入浅出]
  3. java中静态是什么,java中静态和非静态有什么区别
  4. composer Failed to decode zlib stream
  5. 报名|第2期“DI极客说”,揭秘决策AI创新应用带来的行业变革
  6. 恐怖与暴力美学 + 妖魔化:《人皮客栈》观看笔记
  7. UrlRewrite(Url重写技术)
  8. 笔记随笔1(webpack,vue-router,vuex)
  9. Android vlc 分析视频,VLC-Android 对视频流(RTSP)做翻转、旋转
  10. c++ 获取当前时间_ThinkPHP6中获取参数的3种常用方法【总结】
  11. python程序初学者计算器界面_Python初学者,一个简单的计算器程序的数学函数
  12. 【Unity】6.2 在VS2015中调试 C# 脚本
  13. web前端优化--图片优化
  14. 运维工程师可能遇到的面试题
  15. 一些计算机u口无法使用的原因,电脑USB接口不能用的原因大全
  16. win7系统笔记本配置双屏
  17. firefox火狐浏览器flash播放视频
  18. 软件构造复习小结(2)——设计规约(Specification)
  19. 【JS】JavaScript
  20. pandas practice

热门文章

  1. springboot创建子模块时遇到子模块覆盖父模块问题解决
  2. EF关闭自动创建数据库表的方式
  3. sql server 自定义函数
  4. javaweb基础(40)_jdbc框架
  5. DropBox 超实用的免费文件网络同步、备份、分享工具
  6. (IOS)截图Demo
  7. python3.6 websocket异步高并发_在Python3.6上的websocket客户端中侦听传入消息时出现问题...
  8. python如何给定取值范围_python怎么限定函数自变量取值范围
  9. zabbix mysql设置中文乱码_解决zabbix监控因php问题导致图形界面中文乱码方法
  10. centos7安装python3.7.4_Centos7升级Python3.7.4