C# Await
每次提到异步我都选择绕开,感觉深不可测,最近打算看看异步,但又不愿意看书,网上找了几个视频看,发现传智播客的老师讲异步都不是很深入,关键的问题一笔带过,倒是把我弄糊涂了,印象最深刻的是那个老师说的一句话:“在异步函数中,Await之后会自动创建出一个线程”。确实,在控制台程序中是这样,但是在winform中却不都在UI线程中执行,感觉内部隐藏不可告人的秘密,索性自己探索一下。
控制台程序:
static void Main(string[] args)
{
Console.WriteLine("1.同步ThreaID:" + Thread.CurrentThread.ManagedThreadId);
RunTimeAsync();
Console.WriteLine("4.异步执行完毕ThreaID:" + Thread.CurrentThread.ManagedThreadId);
Thread.Sleep();
Console.WriteLine("5.同步延迟完毕ThreaID:" + Thread.CurrentThread.ManagedThreadId);
Console.ReadKey();
}
public async static Task RunTimeAsync()
{
Console.WriteLine("2.进入异步ThreaID:" + Thread.CurrentThread.ManagedThreadId);
await Task.Delay();
Console.WriteLine("3.异步执行后ThreaID:" + Thread.CurrentThread.ManagedThreadId);
}
执行结果:

同样的代码在winform中执行:
winform程序:
private void button1_Click(object sender, EventArgs e)
{
Console.WriteLine("1.同步ThreaID:" + Thread.CurrentThread.ManagedThreadId);
RunTimeAsync();
Console.WriteLine("4.异步执行完毕ThreaID:" + Thread.CurrentThread.ManagedThreadId);
Thread.Sleep();
Console.WriteLine("5.同步延迟完毕ThreaID:" + Thread.CurrentThread.ManagedThreadId); }
public async static Task RunTimeAsync()
{
Console.WriteLine("2.进入异步ThreaID:" + Thread.CurrentThread.ManagedThreadId);
await Task.Delay();
Console.WriteLine("3.异步执行后ThreaID:" + Thread.CurrentThread.ManagedThreadId);
}
执行结果:

前后代码执行结果对比后发现,
1.线程数量,winfrom都是在一个线程里执行,控制台程序是两个线程执行,其中异步操作在新的线程里。
2.执行顺序,winform先执行click中同步操作,再执行异步操作,控制台却相反。
所以await这个关键字到底对异步操作动了什么手脚呢?
在查了一些相关资料后有两个词比较高频:SynchronizationContext 和TaskScheduler,执行代码时,Task不会自己执行,而是将执行内容放到SynchronizationContext 和TaskScheduler执行。
其实await Task.Delay(1000)或者await AsyncFun()(异步操作)等价于以下代码:
Task t = AsyncFun();
var currentContext = SynchronizationContext.Current;
if (null == currentContext)
{
t.ContinueWith((te) => { Run(); }, TaskScheduler.Current);
}
else
{
t.ContinueWith((te) => { Run(); }, TaskScheduler.FromCurrentSynchronizationContext())
}
如果SynchronizationContext为null就在TaskScheduler中执行,如果不为null则在FromCurrentSynchronizationContext中执行。
先了解一下TaskScheduler--------CSDN中注解是:表示一个处理将任务排队到线程中的低级工作的对象。
看一下TaskScheduler的元数据:

TaskScheduler其实就是将Task放到Threadpool执行,通过入队出队的方式排列执行顺序。
根据等价代码我们要区分一下控制台程序与winfrom程序异步操作的执行方式,
我们执行这条代码:var currentContext = SynchronizationContext.Current;
在控制台程序中currentContext 为null
在winform中currentContext 为System.Windows.Forms.WindowsFormsSynchronizationContext
那么控制台程序是在TaskScheduler.Current所在的线程池执行,我们看到控制台的await 异步操作在其他线程中执行一致。
下面分析一下winform的执行方式:
WindowsFormsSynchronizationContext继承自SynchronizationContext,
https://www.cnblogs.com/lzxianren/p/SynchronizationContext.html 参考了这篇文章
里面有两点:
public virtual void Send(SendOrPostCallback d, Object state)
{
d(state);
}
public virtual void Post(SendOrPostCallback d, Object state)
{
ThreadPool.QueueUserWorkItem(new WaitCallback(d), state);
}
在UI线程执行是,会将执行的任务用Send方法放到消息泵中,当时UI中同步执行的操作执行完毕,再执行详细泵中的数据,这就是为什么UI中只在一个线程中执行,而且总是在最后执行。
总结:用 await 语句等待一个任务完成,当该方法在 await 处暂停时,就可以捕捉上下文(context)。如果当前 SynchronizationContext 不为空,这个上下文就是当前SynchronizationContext;如果当前 SynchronizationContext 为空,则这个上下文为当前askScheduler,该方法会在这个上下文中继续运行。一般来说,运行 UI 线程时采用 UI 上下文,处理 ASP.NET 请求时采用 ASP.NET 请求上下文,其他很多情况下则采用线程池上下文。因此,在上面的代码中,每个同步程序块会试图在原始的上下文中恢复运行。如果在 UI线程中调用 ,这个方法的每个同步程序块都将在此 UI 线程上运行。但是,如果在线程池线程中调用,每个同步程序块将在线程池线程上运行。
控制台程序以及类库是在TaskScheduler中调度执行,UI程序则是在SynchronizationContext调度执行。
参考了:https://blog.csdn.net/wlk1229/article/details/81287374
C# Await的更多相关文章
- async & await 的前世今生(Updated)
async 和 await 出现在C# 5.0之后,给并行编程带来了不少的方便,特别是当在MVC中的Action也变成async之后,有点开始什么都是async的味道了.但是这也给我们编程埋下了一些隐 ...
- [.NET] 利用 async & await 的异步编程
利用 async & await 的异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/5922573.html 目录 异步编程的简介 异 ...
- [.NET] 怎样使用 async & await 一步步将同步代码转换为异步编程
怎样使用 async & await 一步步将同步代码转换为异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6079707.html ...
- [.NET] 利用 async & await 进行异步 IO 操作
利用 async & await 进行异步 IO 操作 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6082673.html 序 上次,博主 ...
- [C#] 走进异步编程的世界 - 开始接触 async/await
走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...
- [译] C# 5.0 中的 Async 和 Await (整理中...)
C# 5.0 中的 Async 和 Await [博主]反骨仔 [本文]http://www.cnblogs.com/liqingwen/p/6069062.html 伴随着 .NET 4.5 和 V ...
- await and async
Most people have already heard about the new “async” and “await” functionality coming in Visual Stud ...
- C# await和async
基础阅读:http://www.cnblogs.com/jesse2013/p/async-and-await.html 答疑阅读:http://www.cnblogs.com/heyuquan/ar ...
- C#~异步编程再续~await与async引起的w3wp.exe崩溃-问题友好的解决
返回目录 关于死锁的原因 理解该死锁的原因在于理解await 处理contexts的方式,默认的,当一个未完成的Task 被await的时候,当前的上下文将在该Task完成的时候重新获得并继续执行剩余 ...
- java 线程 Lock 锁使用Condition实现线程的等待(await)与通知(signal)
一.Condition 类 在前面我们学习与synchronized锁配合的线程等待(Object.wait)与线程通知(Object.notify),那么对于JDK1.5 的 java.util.c ...
随机推荐
- PAT Basic 1022 D进制的A+B (20 分)
输入两个非负 10 进制整数 A 和 B (≤),输出 A+B 的 D (1)进制数. 输入格式: 输入在一行中依次给出 3 个整数 A.B 和 D. 输出格式: 输出 A+B 的 D 进制数. 输入 ...
- memset初始化数组的坑
memset函数常被我们用来初始化数组,然而有个坑可能会被我们踩到. 静态数组初始化 一般情形是这样的: #include <cstring> int main() { // 静态数组ar ...
- LeetCode02 - 两数相加(Java 实现)
LeetCode02 - 两数相加(Java 实现) 来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/add-two-numbers 题目描述 ...
- Linux之vi文本编辑器
vi的基本概念 基本上vi可以分为三种状态,分别是命令模式(command mode).输入模式(Insert mode)和末行模式(last line mode),各模式的功能区分如下: 1) 命令 ...
- Web.xml 定制URL
直接上xml里的代码: <!--声明有哪些Servlet--> <servlet> <servlet-name>Book</servlet-name> ...
- SQL 日期转换
), ): :57AMSELECT ), ): ), ): ), ): ), ): ), ): ), ): 06), ): ,06), ): ::46), ): :::827AMSELECT ), ) ...
- electron-vue [Vue warn]: Failed to resolve directive: decorator
electron-vue引入ant-desigin-vue使用ant自定义指令 v-decorator报销 <a-form-item> <a-input v-decorator=&q ...
- Linux帮助文档
Linux当中有许多命令: 在Linux中提供了详细的帮组文档,利用好可以提高使用效率: 1.help参数 大多数命令都可以使用 -h 或 --help 参数来获取该命令的使用方法.参数等信息: ...
- Codevs 1213 解的个数(exgcd)
1213 解的个数 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 已知整数x,y满足如下面的条件: ax+by+c=0 p< ...
- 支持快应用的http网络库-flyio
Fly.js 一个基于Promise的.强大的.支持多种JavaScript运行时的http请求库. 有了它,您可以使用一份http请求代码在浏览器.微信小程序.Weex.Node.React Nat ...