async/await的特殊的地方
一:async如果是用于方法声明里,那么要求这个方法的返回值必须是Task、Task<TResult>、void这三种,而且await出现的地方要求其所在的方法必须是async修饰的方法;
不过void的最好不要用,如果说方法没有返回值用Task,有返回值用Task<TResult>;
先看两段代码:
①
static void Main(string[] args)
{
var task = FooAsync();
// Result的get方法会阻塞当前线程直到task执行完毕
Console.WriteLine(task.Result);
Console.WriteLine("Hello World!");
} static async Task<int> FooAsync()
{
// await task对象可以直接获取task对象执行完毕的返回值,因此await会将Task.Run(..)作为同步块;
int result = await Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕");
return 8;
});
return result;
}
②
static void Main(string[] args)
{
var task = FooAsync();
// Result的get方法会阻塞当前线程直到task执行完毕
Console.WriteLine(task.Result);
Console.WriteLine("Hello World!");
}
// TODO 注意,这个地方去掉了async和里面的await
static Task<int> FooAsync()
{
// 这时候不能用int作为变量类型了
Task<int> result = Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕");
return 8;
});
return result;
}
对于上面的async和await的写法,那么async和await关键字其实可以说一点用处也没有,直接通过Task就能达到完全一模一样的效果;
接着来看另一种写法:
static void Main(string[] args)
{
var task = FooAsync();
// Result的get方法会阻塞当前线程直到task执行完毕
Console.WriteLine(task.Result);
Console.WriteLine("Hello World!");
} /**
* 注意,Task是后台任务(由后台线程执行),而Java的Task,或者说Java默认的线程创建工厂创建的线程是前台线程
*/
static async Task<string> FooAsync()
{
// 这时候不能用int作为变量类型了
Task<int> result = Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕");
return 8;
});
int resultValue = await result;
return "333N" + resultValue; // TODO 注意这里多了转换结果的步骤,用async+await写这种需求很方便,如果只用Task那么就得把Task.Run(...)提升到外部调用的地方才能方便的实现这个功能;
}
这种写法只用Task<TResult>就不容易实现了,至少写起来要多蛮多步骤(需要再手动写个Task<string>对象然后用lambda表达式将await出来的结果转换为string返回出来,然后返回这个new出来的Task<string>对象);
然后再看另外一种写法
static void Main(string[] args)
{
var watch1 = Stopwatch.StartNew();
var watch2 = Stopwatch.StartNew();
Console.WriteLine("Begin");
var task = FooAsync();
watch1.Stop(); Console.WriteLine($"#耗时{watch1.ElapsedMilliseconds}"); // Result的get方法会阻塞当前线程直到task执行完毕
// 这个如果注释掉,那么FooAsync则只阻塞主线程2秒左右,如果不注释下面的代码,那么上面的FooAsync()会先阻塞主线程
//Console.WriteLine(task.Result);
watch2.Stop(); Console.WriteLine($"耗时{watch2.ElapsedMilliseconds}"); Console.WriteLine("End");
} /**
* 注意,Task是后台任务(由后台线程执行),而Java的Task,或者说Java默认的线程创建工厂创建的线程是前台线程(可配置)
*/
static async Task<int> FooAsync()
{
Thread.Sleep(2000); // 这一个会阻塞调用者
Console.WriteLine($"{nameof(FooAsync)}方法先同步耗时2秒");
// 这时候不能用int作为变量类型了
var result = Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕");
return 8;
});
Thread.Sleep(2000); // 重要,这一块不一定会阻塞调用方(await放在Task.Run那里)
Console.WriteLine($"###{nameof(FooAsync)}方法先同步耗时2秒");
// 注意,如果这个await放到上面的Task.Run(..)那里会令这一块的逻辑很奇怪,而且貌似执行是有问题的,都不知道它执行完毕没有
return await result;
}
其实仔细一比较,async和await是可以不需要的,用Task<TResult>就完全可以实现,只不过有部分写法会稍微麻烦一点点而已;await task;就可以用task.Result来代替;
async加await其实就是实现这样一个功能:
有async的方法,是在告诉编译器这个方法内部可能存在异步调用,如Task.Run(...);
然后内部对这个异步调用进行一个await(没有也行但是就失去了async和await带来的糖);
这个糖的最重要的提现在于外部调用这个方法时,如这个方法叫FooAsync(),调用过程为:
var task = FooAsync();
如果接下来的代码没有类似task.Wait()或task.Result这样的等待操作,那么FooAsync()里面的异步部分对于调用者也是异步的(包括await后面的全部代码对于调用者都是异步的,这点很重要,如var result = await Task.Run(...);Thread.Sleep(2000);这个Sleep不会阻塞调用者);
async/await其实是一种类似模板类的技术,有async且内部有await的方法是一种具备不同编译情况的方法(怎么编译要看怎么调用及使用具有async的方法,如调用者有task.Result是一种使用模式,没有task.Result或没有await task或task.Wait..之类的又是另一种模式);
static void Main(string[] args)
{
// async/await其实是一种类似模板类的技术,有async且内部有await的方法是一种具备不同编译情况的方法;
var watch1 = Stopwatch.StartNew();
var watch2 = Stopwatch.StartNew();
Console.WriteLine("Begin");
var task = FooAsync();
watch1.Stop(); Console.WriteLine($"#耗时{watch1.ElapsedMilliseconds}"); // Result的get方法会阻塞当前线程直到task执行完毕
// 这个如果注释掉,那么FooAsync则只阻塞主线程2秒左右,如果不注释下面的代码,那么上面的FooAsync()会先阻塞主线程
//Console.WriteLine(task.Result);
watch2.Stop(); Console.WriteLine($"耗时{watch2.ElapsedMilliseconds}"); Console.WriteLine("End");
// 这里可以通过输出End后立刻按回车或等几秒后按回车进行测试;
Console.ReadKey();
} /**
* 注意,Task是后台任务(由后台线程执行),而Java的Task,或者说Java默认的线程创建工厂创建的线程是前台线程(可配置)
* 甚至编译器将FooAsync()方法编译成了FooAsync`1()和FooAsync`2()两个子方法,然后看调用者是否涉及到同步FooAsync方法结果的操作,有则编译器给调用方调用的是FooAsync`1(),没有则是用的FooAsync`2()方法
*/
static async Task<int> FooAsync()
{
Thread.Sleep(2000); // 这一个会阻塞调用者
Console.WriteLine($"{nameof(FooAsync)}方法先同步耗时2秒");
// 注意,await之后的所有代码是否阻塞调用者取决于调用者怎么用这个方法,如果调用者是await FooAsync(),那么这里之后的所有代码都是同步的
// TODO 所以async和await和Task<TResult>相比多了一个类似开关一样的东西,但是这个开关是编译阶段确定的,有点像模板类用的时候必须给出具体的类型,而这里是具体的用法;
// TODO 编译时如果调用者有task.Result之类的wait操作,那么【编译器】就将FooAsync整个当成一个同步方法,如果没有相关的操作,编译器就将这部分及其后面的代码作为异步方法处理【应该是通过闭包形成一个新的Task对象】
// TODO 这里往下的代码,如果编译器发现调用方不需要同步,那么编译器底层是通过再开一个Task<int>将result和下面的Thread.Sleep(4000)等代码用闭包再包一层返回这个task对象;
var result = await Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕");
return 8;
});
// TODO 注意,这一块是在上面的result执行完后才执行的,即上面先输出 FooAsync方法耗时3秒执行完毕 然后等待4秒后再输出 ###FooAsync方法先同步耗时4秒
Thread.Sleep(4000); // 重要,这一块不一定会阻塞调用方(await放在Task.Run那里),这一块是否阻塞调用者看调用者是否需要对返回的Task对象进行wait,且是编译时决定的;
Console.WriteLine($"###{nameof(FooAsync)}方法先同步耗时4秒");
// 注意,如果这个await放到上面的Task.Run(..)那里会令这一块的逻辑很奇怪,而且貌似执行是有问题的,都不知道它执行完毕没有
return result;
}
来看进一步的测试:
static void Main(string[] args)
{
// async/await其实是一种类似模板类的技术,有async且内部有await的方法是一种具备不同编译情况的方法;
var watch1 = Stopwatch.StartNew();
var watch2 = Stopwatch.StartNew();
Console.WriteLine($"Begin,threadId:{Thread.CurrentThread.ManagedThreadId}");
var task = FooAsync();
watch1.Stop(); Console.WriteLine($"#耗时{watch1.ElapsedMilliseconds}"); // Result的get方法会阻塞当前线程直到task执行完毕
// 这个如果注释掉,那么FooAsync则只阻塞主线程2秒左右,如果不注释下面的代码,那么上面的FooAsync()会先阻塞主线程
//Console.WriteLine(task.Result); // TODO flagN
watch2.Stop(); Console.WriteLine($"耗时{watch2.ElapsedMilliseconds}"); Console.WriteLine("End");
// 这里可以通过输出End后立刻按回车或等几秒后按回车进行测试;
Console.ReadKey();
} /**
* 注意,Task是后台任务(由后台线程执行),而Java的Task,或者说Java默认的线程创建工厂创建的线程是前台线程(可配置)
*/
static async Task<int> FooAsync()
{
Thread.Sleep(2000); // 这一个会阻塞调用者
Console.WriteLine($"{nameof(FooAsync)}方法先同步耗时2秒");
// 注意,await之后的所有代码是否阻塞调用者取决于调用者怎么用这个方法,如果调用者是await FooAsync(),那么这里之后的所有代码都是同步的
// TODO 所以async和await和Task<TResult>相比多了一个类似开关一样的东西,但是这个开关是编译阶段确定的,有点像模板类用的时候必须给出具体的类型,而这里是具体的用法;
// TODO 编译时如果调用者有task.Result之类的wait操作,那么【编译器】就将FooAsync整个当成一个同步方法,如果没有相关的操作,编译器就将这部分及其后面的代码作为异步方法处理【应该是通过闭包形成一个新的Task对象】
// TODO 这里往下的代码,如果编译器发现调用方不需要同步,那么编译器底层是通过再开一个Task<int>将result和下面的Thread.Sleep(4000)等代码用闭包再包一层返回这个task对象;
var result = await Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕,threadId:{Thread.CurrentThread.ManagedThreadId}");
return 8;
});
// TODO 注意,这一块是在上面的result执行完后才执行的,即上面先输出 FooAsync方法耗时3秒执行完毕 然后等待4秒后再输出 ###FooAsync方法先同步耗时4秒
Thread.Sleep(4000); // 重要,这一块不一定会阻塞调用方(await放在Task.Run那里),这一块是否阻塞调用者看调用者是否需要对返回的Task对象进行wait,且是编译时决定的;
Console.WriteLine($"###{nameof(FooAsync)}方法先同步耗时4秒,threadIdNNN:{Thread.CurrentThread.ManagedThreadId}");
return result;
}
经过测试,flagN行无论是否注释,主线程的threadId和后面Task.Run及后面的threadIdNNN的线程id都是不同的,但是后面两个的线程id是一致的,因此编译器底层不是生成了FooAsync`1和FooAsync`2两个方法,而是将await及后面的代码合并为
一块,即FooAsync其实最终是变成这样子的代码(不一定百分百就是这个样子,但是可以解释测试结果):
static async Task<int> FooAsync()
{
Thread.Sleep(2000); // 这一个会阻塞调用者
Console.WriteLine($"{nameof(FooAsync)}方法先同步耗时2秒");
// 注意,await之后的所有代码是否阻塞调用者取决于调用者怎么用这个方法,如果调用者是await FooAsync(),那么这里之后的所有代码都是同步的
// TODO 所以async和await和Task<TResult>相比多了一个类似开关一样的东西,但是这个开关是编译阶段确定的,有点像模板类用的时候必须给出具体的类型,而这里是具体的用法;
// TODO 编译时如果调用者有task.Result之类的wait操作,那么【编译器】就将FooAsync整个当成一个同步方法,如果没有相关的操作,编译器就将这部分及其后面的代码作为异步方法处理【应该是通过闭包形成一个新的Task对象】
// TODO 这里往下的代码,如果编译器发现调用方不需要同步,那么编译器底层是通过再开一个Task<int>将result和下面的Thread.Sleep(4000)等代码用闭包再包一层返回这个task对象;
var result = Task.Run(() =>
{
Thread.Sleep(3000);
Console.WriteLine($"{nameof(FooAsync)}方法耗时3秒执行完毕,threadId:{Thread.CurrentThread.ManagedThreadId}");
var tmp = 8; // TODO 注意,这一块是在上面的result执行完后才执行的,即上面先输出 FooAsync方法耗时3秒执行完毕 然后等待4秒后再输出 ###FooAsync方法先同步耗时4秒
Thread.Sleep(4000); // 重要,这一块不一定会阻塞调用方(await放在Task.Run那里),这一块是否阻塞调用者看调用者是否需要对返回的Task对象进行wait,且是编译时决定的;
Console.WriteLine($"###{nameof(FooAsync)}方法先同步耗时4秒,threadIdNNN:{Thread.CurrentThread.ManagedThreadId}"); return 8;
});
return result;
}
这个result不会阻塞调用者;
async/await的特殊的地方的更多相关文章
- [C#] 走进异步编程的世界 - 开始接触 async/await
走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...
- 实际案例:在现有代码中通过async/await实现并行
一项新技术或者一个新特性,只有你用它解决实际问题后,才能真正体会到它的魅力,真正理解它.也期待大家能够多分享解一些解决实际问题的内容. 在我们遭遇“黑色30秒”问题的过程中,切身体会到了异步的巨大作用 ...
- [C#] .NET4.0中使用4.5中的 async/await 功能实现异
好东西需要分享 原文出自:http://www.itnose.net/detail/6091186.html 在.NET Framework 4.5中添加了新的异步操作库,但是在.NET Framew ...
- 走进异步编程的世界 - 开始接触 async/await
[C#] 走进异步编程的世界 - 开始接触 async/await 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async ...
- async/await 异步编程(转载)
转载地址:http://www.cnblogs.com/teroy/p/4015461.html 前言 最近在学习Web Api框架的时候接触到了async/await,这个特性是.NET 4.5引入 ...
- Python PEP 492 中文翻译——协程与async/await语法
原文标题:PEP 0492 -- Coroutines with async and await syntax 原文链接:https://www.python.org/dev/peps/pep-049 ...
- C# 中 async/await 调用传统 Begin/End 异步方法
最近在改进园子的图片上传程序,希望实现用户上传图片时同时将图片文件保存在三个地方:1)服务器本地硬盘:2)又拍云:3)阿里云OSS.并且在保存时使用异步操作. 对于异步保存到本地硬盘,只需用 Stea ...
- [C#] 走进异步编程的世界 - 开始接触 async/await(转)
原文链接:http://www.cnblogs.com/liqingwen/p/5831951.html 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 ...
- async/await 异步编程
前言 最近在学习Web Api框架的时候接触到了async/await,这个特性是.NET 4.5引入的,由于之前对于异步编程不是很了解,所以花费了一些时间学习一下相关的知识,并整理成这篇博客,如果在 ...
随机推荐
- Android控件使用FragmentTabHost,切换Fragment;
大部分APP的主界面都很类似,要么底部导航的,要么就是侧滑菜单,还有底部导航+侧滑菜单的:底部导航实现大概有几种方式: TabHost+Fragment RadioGroup+Fragment Fra ...
- 面向对象object与constructor
什么是constructor属性?它来自哪里?又将会指向何处? 什么是constructor属性? constructor是构造函数属性. 它来自哪里? 其实constructor属性是来自 prot ...
- bitbucket 上公钥SSH key如何add key并进行项目运用
前提:从sourcetree 添加项目时老是拉取不下来,查到原因是应为bitbucket需要SSH key公钥 目的:公钥相当于你在任何一台电脑只要有公钥授权就可以随时提交代码到服务器 原因: 1.很 ...
- Android 开发 VectorDrawable 矢量图 (一)了解Android矢量图与获取矢量图
VectorDrawable 矢量图 三部曲: Android 开发 VectorDrawable 矢量图 (一)了解Android矢量图与获取矢量图 Android 开发 VectorDrawabl ...
- 彻底解决COM端口被占用(在使用中)问题的办法
今天就遇到这个问题了串口调试的时候发现usb转串口使用的是COM8而串口调试助手里面只有COM1到4,我想去该COM口发现COM1到7都在使用中,找了好多办法都不行,后面在网上找到这篇解决办法的文章, ...
- html常见的块元素和行内元素(特别注意个别块元素不能嵌套其他块元素)
html中常见的块元素:div.p.h1-h6.ul.ol.li.hr.table.pre等 块级元素新开启一行即使是设置了width属性也是独占一行(可设置float浮动属性调整布局).尽可能撑满父 ...
- HTTP协议规定,客户端的编写
HTTP协议是网络应用层协议,建立在TCP/IP协议基础上.HTTP协议基于客户/服务器模式,客户端主动发出HTTP请求,服务器接收HTTP请求,返回HTTP响应结果.HTTP协议对HTTP请求,以及 ...
- cfile fopen fopen_s win10下打开文件失败
vc程序在win10下面使用fopen 打开文件失败,在网上查 了下是因为UAC权限的问题如下: 由于windows vista win7 win8 win 10 添加了UAC权限,所以会导致 在系统 ...
- codeforces 1041A Heist
electronic a.电子的 heist v.抢劫 in ascending order 升序 indice n.标记 device n.装置设备 staff n.职员 in arbitrary ...
- 使用jQuery+huandlebars遍历中if判断
兼容ie8(很实用,复制过来,仅供技术参考,更详细内容请看源地址:http://www.cnblogs.com/iyangyuan/archive/2013/12/12/3471227.html) & ...