【转】C# 彻底搞懂async/await
关键:
异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。
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");}}
【备注】一般来说,直接拼接字符串是一种比较耗性能的手段,如果对字符串拼接有性能要求的话应该使用 StringBuilder。
【注意】每次运行的结果可能不同。不管哪次调试,绝大部分时间都浪费前两次调用(CountCharacters 方法),即在等待网站的响应上。
有人曾幻想着这样提高性能的方法:在调用 A 方法时,不等它执行完,直接执行 B 方法,然后等 A 方法执行完成再处理。
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");}}//这是修改后的代码
【改动分析】只改了几个细节的地方,直接展开代码的话可能看不出来,改动如下:
③然后,Main 方法继续执行三次 ExtraOperation 方法,同时两次 CountCharactersAsync 方法依然在持续工作 。
④t1.Result 和 t2.Result 是指从 CountCharactersAsync 方法调用的 Task<int> 对象取结果,如果还没有结果的话,将阻塞,直有结果返回为止。
二、async/await 结构
同步方法:一个程序调用某个方法,等到其执行完成之后才进行下一步操作。这也是默认的形式。
异步方法:一个程序调用某个方法,在处理完成之前就返回该方法。通过 async/await 我们就可以实现这种类型的方法。
(1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;
(2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;
(3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。
三、What’s 异步方法
异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。
(2)要求:包含 N(N>0) 个 await 表达式(不存在 await 表达式的话 IDE 会发出警告),表示需要异步执行的任务。
(3)返回类型:只能返回 3 种类型(void、Task 和 Task<T>)。Task 和 Task<T> 标识返回的对象会在将来完成工作,表示调用方法和异步方法可以继续执行。
(4)参数:数量不限,但不能使用 out 和 ref 关键字。
(6)其它:匿名方法和 Lambda 表达式也可以作为异步对象;async 是一个上下文关键字;关键字 async 必须在返回类型前。
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相关推荐
- C# 彻底搞懂async/await
前言 Talk is cheap, Show you the code first! private void button1_Click(object sender, EventArgs e) {C ...
- 一次性搞懂JavaScript 执行机制
你是否遭受到这样的恐吓? 你是否有过每个表达式前面都console一遍值去找执行顺序? 看了很多js执行机制的文章似乎都是似懂非懂,到技术面问的时候,理不清思绪.总结了众多文章的例子和精华,希望能帮到 ...
- 异步编程:一次搞懂Promise,async,await
文章目录 前言 一.回调函数 二.Promise 三.错误处理 四.async/await await使用时的陷阱 1 2 3 总结 前言 异步编程允许我们在执行一个长时间任务时,程序不需要进行等待, ...
- 一文看懂async和“await”关键词是如何简化了C#中多线程的开发过程
一文看懂"async"和"await"关键词是如何简化了C#中多线程的开发过程 当我们使用需要长时间运行的方法(即,用于读取大文件或从网络下载大量资源)时,在同 ...
- 一眼看懂promise与async await的区别
// promise方法let p1 = new Promise((resolve,reject) => {setTimeout(() => {resolve('我是p1')},4000) ...
- 在.NET中执行Async/Await的两种错误方法
微信公众号:架构师高级俱乐部 关注可了解更多的编程,架构知识.问题或建议,请公众号留言; 如果你觉得此文对你有帮助,欢迎转发 在.NET中执行异步/等待的两种错误方法 在应用开发中,我们为了提高应用程 ...
- 转:前端 100 问:能搞懂80%的请把简历给我
<前端 100 问:能搞懂80%的请把简历给我> 引言 半年时间,几千人参与,精选大厂前端面试高频 100 题,这就是「壹题」. 在 2019 年 1 月 21 日这天,「壹题」项目正式开 ...
- 这次彻底搞懂 Promise(手写源码多注释篇)
作者:一阵风,一枚只想安静写代码的程序员,来自程序员成长指北交流群 github: https://github.com/yizhengfeng-jj/promise 前言 promise 是 ...
- 【JS】1015- 异步编程的终极解决方案 async/await
早期的回调函数 回调函数我们经常有写到,比如: ajax(url, (res) => {console.log(res); }) 复制代码 但是这种回调函数有一个大缺陷,就是会写出 回调地狱(C ...
最新文章
- 使用模块化编译缩小 apk 体积
- [AaronYang]C#人爱学不学8[事件和.net4.5的弱事件深入浅出]
- java中静态是什么,java中静态和非静态有什么区别
- composer Failed to decode zlib stream
- 报名|第2期“DI极客说”,揭秘决策AI创新应用带来的行业变革
- 恐怖与暴力美学 + 妖魔化:《人皮客栈》观看笔记
- UrlRewrite(Url重写技术)
- 笔记随笔1(webpack,vue-router,vuex)
- Android vlc 分析视频,VLC-Android 对视频流(RTSP)做翻转、旋转
- c++ 获取当前时间_ThinkPHP6中获取参数的3种常用方法【总结】
- python程序初学者计算器界面_Python初学者,一个简单的计算器程序的数学函数
- 【Unity】6.2 在VS2015中调试 C# 脚本
- web前端优化--图片优化
- 运维工程师可能遇到的面试题
- 一些计算机u口无法使用的原因,电脑USB接口不能用的原因大全
- win7系统笔记本配置双屏
- firefox火狐浏览器flash播放视频
- 软件构造复习小结(2)——设计规约(Specification)
- 【JS】JavaScript
- pandas practice
热门文章
- springboot创建子模块时遇到子模块覆盖父模块问题解决
- EF关闭自动创建数据库表的方式
- sql server 自定义函数
- javaweb基础(40)_jdbc框架
- DropBox 超实用的免费文件网络同步、备份、分享工具
- (IOS)截图Demo
- python3.6 websocket异步高并发_在Python3.6上的websocket客户端中侦听传入消息时出现问题...
- python如何给定取值范围_python怎么限定函数自变量取值范围
- zabbix mysql设置中文乱码_解决zabbix监控因php问题导致图形界面中文乱码方法
- centos7安装python3.7.4_Centos7升级Python3.7.4