【转】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");}}

复制代码

 

 

                                            

                                                         图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。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/437389.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LeetCode每日打卡 - 汉明距离总和

有点慢&#xff0c;两层循环也可以完成&#xff0c;就是换个方向&#xff0c;外层遍历32大小的bits数组&#xff0c;里层遍历nums的数字每次右移一位&#xff0c;计算方法类似。 class Solution {public int totalHammingDistance(int[] nums) {//int 是4byte &#xff0c; 每…

python编写代码_用 Python 编写干净、可测试、高质量的代码

用 Python 编写干净、可测试、高质量的代码Noah Gift 2010 年 12 月 20 日发布简介 编写软件是人所承担的最复杂的任务之一。AWK 编程语言和 "K and R C" 的作者之一 Brian Kernigan 在 Software Tools 一书中总结了软件开发的真实性质&#xff0c;他说&#xff0c;“…

阿里云服务器被[kthreaddi]挖矿病毒攻击

首先我根本https://blog.csdn.net/weixin_41599103/article/details/115403332这个博客试了下并没有成功&#xff0c;所以应该是被侵入的程序不一样 先去阿里云里看一下详情 明确告诉了是通过docker被攻击了&#xff0c;先将wordpress容器停止并删除容器和镜像 kill掉进行&am…

python canvas画弧度_python画一朵玫瑰给你

听说 python 还能画画&#xff1f;是的&#xff0c;今天亲手画一朵玫瑰给你。turtle 是 python 中一个强大的绘制图像的库&#xff0c;可以用来绘制各种图像&#xff0c;使用起来很方便。但是需要计算各种角度、距离等&#xff0c;所以要想完成一幅复杂的图像还是有一定的难度的…

通俗彻底解读批处理的延迟变量

先说环境变量 一个批处理可以看成是一个程序&#xff0c;它的环境变量就是自己的全局变量。操作系统可以看成是一个大程序&#xff0c;它的环境变量同样也是全局变量&#xff0c;只不过为了和程序的环境变量区别&#xff0c;就叫做全局环境变量了&#xff0c;既然为了区别&…

LeetCode每日打卡 - 4的幂

不使用循环来今天解题 public boolean isPowerOfFour(int n) {return n > 0 && (n & (n-1)) 0 && n % 31;}首先去做了2的幂这道题&#xff0c;并了解了n&(n-1)的过程&#xff0c;然后思考一下这道题:求二进制中1的个数&#xff0c;理解了n&(n-…

【转】探索c#之Async、Await剖析

阅读目录&#xff1a; 基本介绍基本原理剖析内部实现剖析重点注意的地方总结 基本介绍 Async、Await是net4.x新增的异步编程方式&#xff0c;其目的是为了简化异步程序编写&#xff0c;和之前APM方式简单对比如下。 APM方式&#xff0c;BeginGetRequestStream需要传入回调函数…

时间计算题100道_小学数学专项练习:计算题200道,趁早打印给孩子,期末考试拿100分!...

点击上方「升学的秘诀」关注我们&#xff01;获取更多教育经验、方法、学习资料等&#xff0c;每天中午12点与您相约&#xff01;▼说到我们数学学习过程中最重要的是什么&#xff1f;毋庸置疑那就是我们的一个计算能力&#xff0c;计算可以说是贯穿了我们整个数学学习体系&…

幂等性实现 -接口幂等性

接口幂等性 1.什么是幂等性 对于同一笔业务操作&#xff0c;不管调用多少次&#xff0c;得到的结果都是一样的。 也就是方法调用一次和调用多次产生的额外效果是相同的&#xff0c;他就具有幂等性 2.为什么需要幂等性 在系统高并发的环境下&#xff0c;很有可能因为网络&#…

C 怎么读取Cpp文件_python之调用C加速计算(一)

一、前言python语言是目前比较火的语言&#xff0c;很容易上手&#xff0c;对数据处理也比较友好&#xff0c;可以用几行代码就能进行一些简单的数据处理工作。但是对于稍微大型的数值计算&#xff0c;或者一些涉及到大量循环的数值计算python的计算速度有点让人失望。即使是使…

【转】刨根究底字符编码【2.0版】(3):字符编码的由来、演变与ASCII码

为什么需要字符编码 1. 计算机一开始发明出来时是用来解决数字计算问题的&#xff0c;后来人们发现&#xff0c;计算机还可以做更多的事&#xff0c;例如文本处理。 但计算机其实挺“笨”的&#xff0c;它只“认识”010110111000…这样由0和1两个数字组成的二进制数字&#…

JS创建对象的模式介绍

转自http://www.cnblogs.com/asqq/archive/2013/02/01/3194993.html

matplotlib的优点_超详细matplotlib基础介绍!!!

(给Python开发者加星标&#xff0c;提升Python技能)来源&#xff1a;逐梦erhttps://zhumenger.blog.csdn.net/article/details/106530281【导语】&#xff1a;出色的数据可视化&#xff0c;会让你的数据分析等工作锦上添花&#xff0c;让人印(升)象(职)深(加)刻(薪)。matplotli…

【转】WPF PRISM开发入门一( 初始化PRISM WPF程序)

这篇博客将介绍在WPF项目中引入PRISM框架进行开发的一些基础知识。目前最新的PRISM的版本是Prism 6.1.0&#xff0c;可以在Github上获取PRISM的源码。这个系列的博客将选择PRISM 4.1版本来讲解。可以从微软官网上下载到PRISM 4.1相关内容。将下载下来的文件解压开&#xff1a; …

截屏悬浮软件_功能强大,却小巧的录屏软件,不在错过你的王者时刻

看看录屏是一款操作简单。功能强大的录屏软件。他可以设置你录制视频的一个分辨率&#xff0c;帧率以及录制屏幕方向&#xff0c;非常方便&#xff0c;用户将手机摇一摇就可以控制开启和停止录屏&#xff0c;高效录制精彩瞬间&#xff0c;在录制游戏视频的时候也可以做到不掉帧…

公司用的非标普通自动化用单片机还是plc_PLC的介绍

PLC又叫可编程控制器&#xff0c;一开始是替代传统接触器的一个东西。随着人工价格不断的上涨&#xff0c;自动化的设备会越来越普及。自动化不再是大企业才用的起的东西 &#xff0c;各种多元化小型自动化设备进入了普通小企业甚至家庭作坊。PLC其实是单片机开发出来的一种工业…

比较文本差异的工具_Linux 开发的五大必备工具 | Linux 中国

Linux 已经成为工作、娱乐和个人生活等多个领域的支柱&#xff0c;人们已经越来越离不开它。在 Linux 的帮助下&#xff0c;技术的变革速度超出了人们的想象&#xff0c;Linux 开发的速度也以指数规模增长。因此&#xff0c;越来越多的开发者也不断地加入开源和学习 Linux 开发…

【转】C# 动态对象(dynamic)的用法

说到正确用法&#xff0c;那么首先应该指出一个错误用法&#xff1a; 常有人会拿var这个关键字来和dynamic做比较。实际上&#xff0c;var和dynamic完全是两个概念&#xff0c;根本不应该放在一起做比较。var实际上是编译期抛给我们的“语法糖”&#xff0c;一旦被编译&#x…

关于prototype使用位置问题的讨论

问题贴&#xff1a;http://bbs.csdn.net/topics/390446362 new四部曲&#xff1a; &#xff08;1&#xff09;创建一个新的对象&#xff0c;并让函数的 this 指针指向它&#xff1b; &#xff08;2&#xff09;将函数的 prototype 对象的所有成员都赋给这个新对象&#xff0c…