await之后的线程问题
之前看了园子里的一篇文章「async & await的前世今生」,收益颇多。而其中有句话被博主特意用红色标注,所以留意多看了几眼,「await 之后不会开启新的线程(await 从来不会开启新的线程)」。在MSDN上找到的相关资料也佐证了其正确性——The async and await keywords don't cause additional threads to be created. Async methods don't require multithreading because an async method doesn't run on its own thread. The method runs on the current synchronization context and uses time on the thread only when the method is active.(async 和 await 关键字不会导致创建其他线程。 因为异步方法不会在其自身线程上运行,因此它不需要多线程。 只有当方法处于活动状态时,该方法将在当前同步上下文中运行并使用线程上的时间。)
再建立一个Windows Forms应用工程,写点代码更形象地说明问题:
private void Form1_Load(object sender, EventArgs e)
{
PrintDataAsync();
Debug.Print("three");
} private async void PrintDataAsync()
{
Task<int> result = CalculateDataAsync();
Debug.Print("second");
int data = await result;
Debug.Print("last:" + data);
} private async Task<int> CalculateDataAsync()
{
Debug.Print(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
Debug.Print("first");
int result = 0;
for (int i = 0; i < 10; i++)
{
result += i;
}
await Task.Delay(1000);
Debug.Print("four");
Debug.Print(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
return result;
};
程序的结果如预期的一样,Output窗口中可以看到以下内容:
8 : False
first
second
three
four
8 : False
last:45
await之前的ManagedThreadId值与之后的ManagedThreadId值一致,IsThreadPoolThread始终是False,说明当前线程没有发生改变,也没有产生新的线程。
但如果建立的是Console应用工程,结果就不同了。
static void Main(string[] args)
{
PrintDataAsync();
Console.WriteLine("three");
Console.Read();
} private static async void PrintDataAsync()
{
Task<int> result = CalculateDataAsync();
Console.WriteLine("second");
int data = await result;
Console.WriteLine("last:" + data);
} private static async Task<int> CalculateDataAsync()
{
Console.WriteLine(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
Console.WriteLine("first");
int result = 0;
for (int i = 0; i < 10; i++)
{
result += i;
}
await Task.Delay(1000);
Console.WriteLine("four");
Console.WriteLine(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
return result;
}
这段代码的执行结果:
8 : False
first
second
three
four
10 : True
last:45
ManagedThreadId在await之后发生了变化,IsThreadPoolThread也变为了True,说明不是同一个线程。
为什么会这样?再看一下MSDN中描述——「The method runs on the current synchronization context and uses time on the thread only when the method is active」,这里涉及到SynchronizationContext对象的使用。
在Windows Forms工程代码中加入 Debug.Print(SynchronizationContext.Current.ToString()); 检测代码,其输出是System.Windows.Forms.WindowsFormsSynchronizationContext。
而如果在Console工程中加入类似的检测代码 Console.WriteLine(SynchronizationContext.Current.ToString()); 则会抛出空引用异常,因为SynchronizationContext.Current在Console工程中的值为null。
又从MSDN Magazine找到SynchronizationContext相关的文章,其中有介绍到:By convention, if a thread’s current SynchronizationContext is null, then it implicitly has a default SynchronizationContext.(根据惯例,如果一个线程的当前 SynchronizationContext 为 null,那么它隐式具有一个默认 SynchronizationContext。)The default SynchronizationContext is applied to ThreadPool threads unless the code is hosted by ASP.NET.(默认 SynchronizationContext 应用于 ThreadPool 线程,除非代码由 ASP.NET 承载。)
这里提到了APS.NET,所以再建个Web Forms应用工程用于验证:
protected void Page_Load(object sender, EventArgs e)
{
PrintDataAsync();
Debug.Print("three");
} private async void PrintDataAsync()
{
Debug.Print(SynchronizationContext.Current.ToString());
Task<int> result = CalculateDataAsync();
Debug.Print("second");
int data = await result;
Debug.Print("last:" + data);
} private async Task<int> CalculateDataAsync()
{
Debug.Print(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
Debug.Print("first");
int result = 0;
for (int i = 0; i < 10; i++)
{
result += i;
}
await Task.Delay(1000);
Debug.Print("four");
Debug.Print(string.Format("{0} : {1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread));
return result;
}
输出结果:
System.Web.AspNetSynchronizationContext
8 : True
first
second
three
four
9 : True
last:45
ManagedThreadId值发生改变,IsThreadPoolThread始终是True,SynchronizationContext.Current值为System.Web.AspNetSynchronizationContext。
由三次试验及相关资料可以得出结论,await之后的线程依据SynchronizationContext在不同环境中的不同定义而产生不同的结果。所以「await 之后不会开启新的线程(await 从来不会开启新的线程)」的肯定句式改成「await 之后会开启新的线程吗? Maybe」这样的句式更加合适些。
最后补充一点,若是把第一个Windows Forms工程的代码 await Task.Delay(); 改成 await Task.Delay().ConfigureAwait(false); 的话,则可以得到第二个Console工程同样的结果。
await之后的线程问题的更多相关文章
- C# 中await前后执行线程的问题
悬赏园豆:20 [已解决问题] 浏览: 1763次 解决于 2018-08-15 22:43 今天有点疑惑就写了个测试的代码,发现控制台和Winform中不一样 比如: 控制台: ...Main( ...
- 浅谈C#中的 async await 以及对线程相关知识的复习
C#5.0以后新增了一个语法糖,那就是异步方法async await,之前对线程,进程方面的知识有过较为深入的学习,大概知道这个概念,我的项目中实际用到C#异步编程的场景比较少,就算要用到一般也感觉T ...
- java 线程 Lock 锁使用Condition实现线程的等待(await)与通知(signal)
一.Condition 类 在前面我们学习与synchronized锁配合的线程等待(Object.wait)与线程通知(Object.notify),那么对于JDK1.5 的 java.util.c ...
- async与await线程分配研究
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- C#语法——泛型的多种应用 C#语法——await与async的正确打开方式 C#线程安全使用(五) C#语法——元组类型 好好耕耘 redis和memcached的区别
C#语法——泛型的多种应用 本篇文章主要介绍泛型的应用. 泛型是.NET Framework 2.0 版类库就已经提供的语法,主要用于提高代码的可重用性.类型安全性和效率. 泛型的定义 下面定义了 ...
- 四、线程同步之Lock和Condition
Lock同步锁 Lock 在jdk1.5 提供了Lock以便执行同步操作,和synchronized不同的是Lock提供了显示的方法获取锁和释放锁.Lock提供了以下几个方法,请求和释放锁: voi ...
- C#~异步编程再续~await与async引起的w3wp.exe崩溃
返回目录 最近怪事又开始发生了,IIS的应用程序池无做挂掉,都指向同一个矛头,async,threadPool,Task,还有一个System.NullReferenceException,所以这些都 ...
- async 和 await小结
三大返回值: 返回类型 - Task<TResult> 返回类型 - Task 返回类型 - void 当你添加 async 关键字后,需要返回一个将用于后续操作的对象,请使用 Task& ...
- Java线程间通信方式剖析——Java进阶(四)
原创文章,同步发自作者个人博客,转载请在文章开头处以超链接注明出处 http://www.jasongj.com/java/thread_communication/ CountDownLatch C ...
随机推荐
- LSD-SLAM深入学习(4)-非ROS改造
前言 没错,距离上一次博客的发布已经俩月了,今天是圣诞节,圣诞快乐. 在前几篇中已经完成了ROS下面的一系列操作.如有任何问题,feel free to contact me at robotsmin ...
- SSM框架——使用MyBatis Generator自动创建代码
版权声明:本文为博主原创文章,未经博主允许不得转载. 这两天需要用到MyBatis的代码自动生成的功能,由于MyBatis属于一种半自动的ORM框架,所以主要的工作就是配置Mapping映射文件,但是 ...
- 用Action的属性接受参数
版本, @Override是Java5的元数据,自动加上去的一个标志,告诉你说下面这个方法是从父类/接口 继承过来的,需要你重写一次,这样就可以方便你阅读,也不怕会忘记@Override是伪代码,表示 ...
- Openfire 集群探究_压测情况
一.(测试时间:20151220 - 下午14:00-17:00) windows环境 第一次测试,运行中发现在eclipse环境下.4w个连接耗尽了约1G的内存. 另外:服务本来可用的,压测客户端强 ...
- Android框架之AndroidAnnotations基础
一:开源网址 https://github.com/excilys/androidannotations/wiki 二:AndroidAnnotation特点 (1)依赖注入 可以注入 views, ...
- Linux_arm驱动之按键模拟脉冲实现定时器的精确计时
/***************************************************************** 内核驱动部分button_ker.c ************** ...
- 浅谈SDN和NFV之间的关系
一个行业固定设备的折旧周期很长,任何变革的发生都绝非易事,但是网络却一次性面临两项革新--软件定义网络(SDN)和网络功能虚拟化(NFV),在变革网络的过程中,二者若想取得成功可能会依赖彼此的技术,或 ...
- 怎样防止重复发送 Ajax 请求?
著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处.作者:长天之云链接:http://www.zhihu.com/question/19805411/answer/15465427来源 ...
- 关于Java深clone 的例子学习
之前http://www.cnblogs.com/lhppom/p/4857702.html里有提到关于Java的深克隆的学习,深浅区别就是在于仅复制对象引用和复制对象引用所指向的对象,最近在看< ...
- Robot Framework安装教程
第一步:安装Python,安装的版本是python-2.7.9.amd64.msi 安装教程详见地址:http://jingyan.baidu.com/article/c910274be14d64cd ...