第十五节:深入理解async和await的作用及各种适用场景和用法
一. 同步VS异步
1. 同步 VS 异步 VS 多线程
2. 常见的异步方法(都以Async结尾)
① HttpClient类:PostAsync、PutAsync、GetAsync、DeleteAsync
② EF中DbContext类:SaveChangesAsync
③ 文件相关中的:WriteLineAsync

3. 引入异步方法的背景
比如我在后台要向另一台服务器中获取中的2个接口获取信息,然后将两个接口的信息拼接起来,一起输出,接口1耗时3s,接口2耗时5s,
① 传统的同步方式:
需要的时间大约为:3s + 5s =8s, 如下面 【案例1】
先分享一个同步请求接口的封装方法,下同。
public class HttpService
{
/// <summary>
/// 后台跨域请求发送代码
/// </summary>
/// <param name="url">eg:http://ac.guojin.org/jeesite/regist/saveAppAgentAccount </param>
///<param name="postData"></param>
/// 参数格式(手拼Json) string postData = "{\"name\":\"" + vip.comName + "\",\"shortName\":\"" + vip.shortName + + "\"}";
/// <returns></returns>
public static string PostData(string postData, string url)
{
HttpWebRequest req = (HttpWebRequest)WebRequest.Create(url);//后台请求页面
Encoding encoding = Encoding.GetEncoding("utf-8");//注意页面的编码,否则会出现乱码
byte[] requestBytes = encoding.GetBytes(postData);
req.Method = "POST";
req.ContentType = "application/json";
req.ContentLength = requestBytes.Length;
Stream requestStream = req.GetRequestStream();
requestStream.Write(requestBytes, , requestBytes.Length);
requestStream.Close();
HttpWebResponse res = (HttpWebResponse)req.GetResponse();
StreamReader sr = new StreamReader(res.GetResponseStream(), System.Text.Encoding.GetEncoding("utf-8"));
string backstr = sr.ReadToEnd();//可以读取到从页面返回的结果,以数据流的形式。
sr.Close();
res.Close(); return backstr;
}
然后在分享服务上的耗时操作,下同。
/// <summary>
/// 耗时方法 耗时3s
/// </summary>
/// <returns></returns>
public ActionResult GetMsg1()
{
Thread.Sleep();
return Content("GetMsg1"); } /// <summary>
/// 耗时方法 耗时5s
/// </summary>
/// <returns></returns>
public ActionResult GetMsg2()
{
Thread.Sleep();
return Content("GetMsg2"); }
下面是案例1代码
#region 案例1(传统同步方式 耗时8s左右)
{
Stopwatch watch = Stopwatch.StartNew();
Console.WriteLine("开始执行"); string t1 = HttpService.PostData("", "http://localhost:2788/Home/GetMsg1");
string t2 = HttpService.PostData("", "http://localhost:2788/Home/GetMsg2"); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1},{t2}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

② 开启新线程分别执行两个耗时操作
需要的时间大约为:Max(3s,5s) = 5s ,如下面【案例2】
#region 案例2(开启新线程分别执行两个耗时操作 耗时5s左右)
{
Stopwatch watch = Stopwatch.StartNew();
Console.WriteLine("开始执行"); var task1 = Task.Run(() =>
{
return HttpService.PostData("", "http://localhost:2788/Home/GetMsg1");
}); var task2 = Task.Run(() =>
{
return HttpService.PostData("", "http://localhost:2788/Home/GetMsg2");
}); Console.WriteLine("我是主业务");
//主线程进行等待
Task.WaitAll(task1, task2);
Console.WriteLine($"{task1.Result},{task2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

既然②方式可以解决同步方法串行耗时间的问题,但这种方式存在一个弊端,一个业务中存在多个线程,且需要对线程进行管理,相对麻烦,从而引出了异步方法。
这里的异步方法 我 特指:系统类库自带的以async结尾的异步方法。
③ 使用系统类库自带的异步方法
需要的时间大约为:Max(3s,5s) = 5s ,如下面【案例3】
#region 案例3(使用系统类库自带的异步方法 耗时5s左右)
{
Stopwatch watch = Stopwatch.StartNew();
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
Console.WriteLine("开始执行");
//执行业务
var r1 = http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
var r2 = http.PostAsync("http://localhost:2788/Home/GetMsg2", httpContent);
Console.WriteLine("我是主业务"); //通过异步方法的结果.Result可以是异步方法执行完的结果
Console.WriteLine(r1.Result.Content.ReadAsStringAsync().Result);
Console.WriteLine(r2.Result.Content.ReadAsStringAsync().Result); watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

PS:通过 .Result 来获取异步方法执行完后的结果。
二. 利用async和await封装异步方法
1. 首先要声明几点:
① async和await关键字是C# 5.0时代引入的,它是一种异步编程模型
② 它们本身并不创建新线程,但我可以在自行封装的async中利用Task.Run开启新线程
③ 利用async关键字封装的方法中如果写全部都是一些串行业务, 且不用await关键字,那么即使使用async封装,也并没有什么卵用,并起不了异步方法的作用。
需要的时间大约为:3s + 5s =8s, 如下面 【案例4】,并且封装的方法编译器会提示:“缺少关键字await,将以同步的方式调用,请使用await运算符等待非阻止API或Task.Run的形式”(PS:非阻止API指系统类库自带的以Async结尾的异步方法)
//利用async封装同步业务的方法
private static async Task<string> NewMethod5Async()
{
Thread.Sleep();
//其它同步业务
return "Msg1";
}
private static async Task<string> NewMethod6Async()
{
Thread.Sleep();
//其它同步业务
return "Msg2";
}
#region 案例4(async关键字封装的方法中如果写全部都是一些串行业务 耗时8s左右)
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行"); Task<string> t1 = NewMethod5Async();
Task<string> t2 = NewMethod6Async(); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1.Result},{t2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

观点结论1:从上面③中可以得出一个结论,async中必须要有await运算符才能起到异步方法的作用,且await 运算符只能加在 系统类库默认提供的异步方法或者新线程(如:Task.Run)前面。
如:下面【案例5】 和 【案例6】需要的时间大约为:Max(3s,5s) = 5s
// 将系统类库提供的异步方法利用async封装起来
private static async Task<String> NewMethod1Async()
{
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
return r1.Content.ReadAsStringAsync().Result;
}
private static async Task<String> NewMethod2Async()
{
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg2", httpContent);
return r1.Content.ReadAsStringAsync().Result;
} //将await关键字加在新线程的前面
private static async Task<string> NewMethod3Async()
{
var msg = await Task.Run(() =>
{
return HttpService.PostData("", "http://localhost:2788/Home/GetMsg1");
});
return msg;
}
private static async Task<string> NewMethod4Async()
{
var msg = await Task.Run(() =>
{
return HttpService.PostData("", "http://localhost:2788/Home/GetMsg2");
});
return msg;
}
#region 案例5(将系统类库提供的异步方法利用async封装起来 耗时5s左右)
//并且先输出“我是主业务”,证明t1和t2是并行执行的,且不阻碍主业务
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行");
Task<string> t1 = NewMethod1Async();
Task<string> t2 = NewMethod2Async(); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1.Result},{t2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

#region 案例6(将新线程利用async封装起来 耗时5s左右)
//并且先输出“我是主业务”,证明t1和t2是并行执行的,且不阻碍主业务
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行");
Task<string> t1 = NewMethod3Async();
Task<string> t2 = NewMethod4Async(); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1.Result},{t2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

2. 几个规则和约定
① async封装的方法中,可以有多个await,这里的await代表等待该行代码执行完毕。
② 我们通常自己封装的方法也要以Async结尾,方便识别
③ 异步返回类型主要有三种:Task<T> 、Task、Void
3. 测试得出其他几个结论
① 如果async封装的异步方法里既有同步业务又有异步业务(开启新线程或者系统类库提供异步方法),那么同步方法那部分的时间在调用的时候是会阻塞主线程的,即主线程要等待这部分同步业务执行完才能往下执行。
如【案例7】 耗时:同步操作之和 2s+2s + Max(3s,5s)=9s;
//同步耗时操作和异步方法同时封装
private static async Task<String> NewMethod7Async()
{
//调用异步方法之前还有一个耗时操作
Thread.Sleep(); //下面的操作耗时3s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
return r1.Content.ReadAsStringAsync().Result;
}
private static async Task<String> NewMethod8Async()
{
//调用异步方法之前还有一个耗时操作
Thread.Sleep(); //下面的操作耗时5s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg2", httpContent);
return r1.Content.ReadAsStringAsync().Result;
}
#region 案例7(既有普通的耗时操作,也有系统本身的异步方法,耗时9s左右)
//且大约4s后才能输出 “我是主业务”,证明同步操作Thread.Sleep(2000); 阻塞主线程
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行");
Task<string> t1 = NewMethod7Async();
Task<string> t2 = NewMethod8Async(); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1.Result},{t2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

证明:async封装的异步方法里的同步业务的时间会阻塞主线程,再次证明 await只能加在 非阻止api和开启新线程的前面
② 如果封装的异步方法中存在等待的问题,而且不能阻塞主线程(不能用Thread.Sleep) , 这个时候可以用Task.Delay,并在前面加await关键字
如【案例8】 耗时:Max(2+3 , 5+2)=7s
//利用Task.Delay(2000);等待
private static async Task<String> NewMethod11Async()
{
//调用异步方法之前需要等待2s
await Task.Delay(); //下面的操作耗时3s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
return r1.Content.ReadAsStringAsync().Result;
} private static async Task<String> NewMethod12Async()
{
//调用异步方法之前需要等待2s
await Task.Delay(); //下面的操作耗时5s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg2", httpContent);
return r1.Content.ReadAsStringAsync().Result;
}
#region 案例8(利用Task.Delay执行异步方法的等待操作)
//结果是7s,且马上输出“我是主业务”,说明Task.Delay(),不阻塞主线程。
{
Stopwatch watch = Stopwatch.StartNew();
Console.WriteLine("开始执行");
Task<string> t1 = NewMethod11Async();
Task<string> t2 = NewMethod12Async(); Console.WriteLine("我是主业务");
Console.WriteLine($"{t1.Result},{t2.Result}");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

三. 异步方法返回类型
1. Task<T>, 处理含有返回值的异步方法,通过 .Result 等待异步方法执行完,且获取到返回值。
2. Task:调用方法不需要从异步方法中取返回值,但是希望检查异步方法的状态,那么可以选择可以返回 Task 类型的对象。不过,就算异步方法中包含 return 语句,也不会返回任何东西。
如【案例9】
//返回值为Task的方法
private static async Task NewMethod9Async()
{
//下面的操作耗时3s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
Console.WriteLine("NewMethod9Async执行完成");
}
#region 案例9(返回值为Task的异步方法)
//结果是5s,说明异步方法和主线程的同步方法 在并行执行
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行");
Task t = NewMethod9Async(); Console.WriteLine($"{nameof(t.Status)}: {t.Status}"); //任务状态
Console.WriteLine($"{nameof(t.IsCompleted)}: {t.IsCompleted}"); //任务完成状态标识
Console.WriteLine($"{nameof(t.IsFaulted)}: {t.IsFaulted}"); //任务是否有未处理的异常标识 //执行其他耗时操作,与此同时NewMethod9Async也在工作
Thread.Sleep(); Console.WriteLine("我是主业务"); t.Wait(); Console.WriteLine($"{nameof(t.Status)}: {t.Status}"); //任务状态
Console.WriteLine($"{nameof(t.IsCompleted)}: {t.IsCompleted}"); //任务完成状态标识
Console.WriteLine($"{nameof(t.IsFaulted)}: {t.IsFaulted}"); //任务是否有未处理的异常标识 Console.WriteLine($"所有业务执行完成了");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

PS:对于Task返回值的异步方法,可以调用Wait(),等 待该异步方法执行完,他和await不同,await必须出现在async关键字封装的方法中。
3. void:调用异步执行方法,不需要做任何交互
如【案例10】
//返回值是Void的方法
private static async void NewMethod10Async()
{
//下面的操作耗时5s
HttpClient http = new HttpClient();
var httpContent = new StringContent("", Encoding.UTF8, "application/json");
//执行业务,假设这里主需要请求,不需要做任何交互
var r1 = await http.PostAsync("http://localhost:2788/Home/GetMsg1", httpContent);
Console.WriteLine("NewMethod10Async执行完成");
}
#region 案例10(返回值为Void的异步方法)
//结果是5s,说明异步方法和主线程的同步方法 在并行执行
{
Stopwatch watch = Stopwatch.StartNew(); Console.WriteLine("开始执行");
NewMethod10Async(); //执行其他耗时操作,与此同时NewMethod9Async也在工作
Thread.Sleep(); Console.WriteLine("我是主业务"); Console.WriteLine($"所有业务执行完成了");
watch.Stop();
Console.WriteLine($"耗时:{watch.ElapsedMilliseconds}");
}
#endregion

四. 几个结论
1. 异步方法到底开不开起新线程?
异步和等待关键字不会导致其他线程创建。 因为异步方法本身并不会运行的线程,异步方法不需要多线程。 只有 + 当方法处于活动状态,则方法在当前同步上下文中运行并使用在线程的时间。 可以使用 Task.Run 移动 CPU 工作移到后台线程,但是,后台线程不利于等待结果变得可用处理。(来自MSDN原话)
2. async和await是一种异步编程模型,它本身并不能开启新线程,多用于将一些非阻止API或者开启新线程的操作封装起来,使其调用的时候像同步方法一样使用。
下面补充博客园dudu的解释,方便大家理解。

五. 参考资料
1. 反骨仔:http://www.cnblogs.com/liqingwen/p/5831951.html
http://www.cnblogs.com/liqingwen/p/5844095.html
2. MSDN:https://msdn.microsoft.com/library/hh191443(vs.110).aspx
PS:如果你想了解多线程的其他知识,请移步:那些年我们一起追逐的多线程(Thread、ThreadPool、委托异步调用、Task/TaskFactory、Parallerl、async和await)
!
- 作 者 : Yaopengfei(姚鹏飞)
- 博客地址 : http://www.cnblogs.com/yaopengfei/
- 声 明1 : 本人才疏学浅,用郭德纲的话说“我是一个小学生”,如有错误,欢迎讨论,请勿谩骂^_^。
- 声 明2 : 原创博客请在转载时保留原文链接或在文章开头加上本人博客地址,如需代码请加我QQ:604649488 (备注:评论的博客名)
第十五节:深入理解async和await的作用及各种适用场景和用法的更多相关文章
- Ext JS学习第十六天 事件机制event(一) DotNet进阶系列(持续更新) 第一节:.Net版基于WebSocket的聊天室样例 第十五节:深入理解async和await的作用及各种适用场景和用法 第十五节:深入理解async和await的作用及各种适用场景和用法 前端自动化准备和详细配置(NVM、NPM/CNPM、NodeJs、NRM、WebPack、Gulp/Grunt、G
code&monkey Ext JS学习第十六天 事件机制event(一) 此文用来记录学习笔记: 休息了好几天,从今天开始继续保持更新,鞭策自己学习 今天我们来说一说什么是事件,对于事件 ...
- 深入理解async和await的作用及各种适用场景和用法
https://www.cnblogs.com/yaopengfei/archive/2018/07/02/9249390.html https://www.cnblogs.com/xianyudot ...
- 第四百一十五节,python常用排序算法学习
第四百一十五节,python常用排序算法学习 常用排序 名称 复杂度 说明 备注 冒泡排序Bubble Sort O(N*N) 将待排序的元素看作是竖着排列的“气泡”,较小的元素比较轻,从而要往上浮 ...
- centos Linux系统日常管理2 tcpdump,tshark,selinux,strings命令, iptables ,crontab,TCP,UDP,ICMP,FTP网络知识 第十五节课
centos Linux系统日常管理2 tcpdump,tshark,selinux,strings命令, iptables ,crontab,TCP,UDP,ICMP,FTP网络知识 第十五节课 ...
- 大白话5分钟带你走进人工智能-第十五节L1和L2正则几何解释和Ridge,Lasso,Elastic Net回归
第十五节L1和L2正则几何解释和Ridge,Lasso,Elastic Net回归 上一节中我们讲解了L1和L2正则的概念,知道了L1和L2都会使不重要的维度权重下降得多,重要的维度权重下降得少,引入 ...
- 第三百八十五节,Django+Xadmin打造上线标准的在线教育平台—登录功能实现,回填数据以及错误提示html
第三百八十五节,Django+Xadmin打造上线标准的在线教育平台—登录功能实现 1,配置登录路由 from django.conf.urls import url, include # 导入dja ...
- 第三百七十五节,Django+Xadmin打造上线标准的在线教育平台—创建课程机构app,在models.py文件生成3张表,城市表、课程机构表、讲师表
第三百七十五节,Django+Xadmin打造上线标准的在线教育平台—创建课程机构app,在models.py文件生成3张表,城市表.课程机构表.讲师表 创建名称为app_organization的课 ...
- 第三百六十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—elasticsearch(搜索引擎)的基本查询
第三百六十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—elasticsearch(搜索引擎)的基本查询 1.elasticsearch(搜索引擎)的查询 elasticsearch是功能 ...
- 第三百五十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—scrapy信号详解
第三百五十五节,Python分布式爬虫打造搜索引擎Scrapy精讲—scrapy信号详解 信号一般使用信号分发器dispatcher.connect(),来设置信号,和信号触发函数,当捕获到信号时执行 ...
随机推荐
- SAP CRM 集类型(Set Type)与产品层次(Product Hierarchy)
本文是产品与对象相关的部分SAP文档的翻译,不包含配置部分. 本文链接:https://www.cnblogs.com/hhelibeb/p/10112723.html 1,对象(Objects) 对 ...
- Springboot整合Ehcache缓存
Pom.xml导包 <!-- ehcache --> <dependency> <groupId>org.springframework.boot</grou ...
- Python开发【内置模块篇】
动态导入模块 动态导入模块 导入一个库名为字符串的 module_t = __import__('m1.t') print (module_t) #m1 import importlib m=impo ...
- Redis内存优化memory-optimization
https://redis.io/topics/memory-optimization 官方文档 一.特殊编码: 自从Redis 2.2之后,很多数据类型都可以通过特殊编码的方式来进行存储空间的优化 ...
- 【Atcoder Grand Contest 011 F】Train Service Planning
题意:给\(n+1\)个站\(0,\dots,n\),连续的两站\(i-1\)和\(i\)之间有一个距离\(A_i\),其是单行(\(B_i=1\))或双行(\(B_i=2\)),单行线不能同时有两辆 ...
- C语言的3种参数传递方式
参数传递,是在程序运行过程中,实际参数就会将参数值传递给相应的形式参数,然后在函数中实现对数据处理和返回的过程,方法有3种方式 值传递 地址传递 引用传递 tips: 被调用函数的形参只有函数被调用时 ...
- 迷茫<第二篇:回到老家湖南长沙>
2014年8月初,我买了回老家的火车票,当时没有买到坐票,卧铺贵了买不起,所以我就选择了站票,准备站回老家.我现在还记得我当时买的是T1列火车,北京西站到长沙火车站,全程16个小时.当时我就在火车上站 ...
- 六招轻松搞定你的CentOS系统安全加固
Redhat是目前企业中用的最多的一类Linux,而目前针对Redhat攻击的黑客也越来越多了.我们要如何为这类服务器做好安全加固工作呢? 一. 账户安全 1.1 锁定系统中多余的自建帐号 检查方 ...
- PS制作简洁漂亮的立体抽丝文字
一.新建一个800*600px文档,并将Background图层创建一个副本,将其命名为Background_copy. 二.双击Background_copy图层,勾选渐变叠加,并设定以下数值 勾选 ...
- 3 数据分析之Numpy模块(2)
数组函数 通用元素级数组函数通用函数(即ufunc)是一种对ndarray中的数据执行元素级的运算.我们可以将其看做是简单的函数(接收一个或多个参数,返回一个或者多个返回值). 常用一元ufunc: ...