这次来理解一下异步方法与线程之间的关系

新建一个控制台程序 代码如下

static void Main(string[] args)
{
Console.WriteLine("\n进入Main()方法,执行线程ID:{0},来自线程池?{1},是背景线程?{2}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread, Thread.CurrentThread.IsBackground);
TestDoWorkAsync();
Console.WriteLine("\n返回Main()方法,等待用户敲击任意键退出,执行线程ID:{0},来自线程池?{1},是背景线程?{2}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread, Thread.CurrentThread.IsBackground);
Console.ReadKey();
} private async static void TestDoWorkAsync()
{
Console.WriteLine("\n进入TestDoWorkAsync()方法,await语句之前的代码执行线程ID:{0},来自线程池?{1},是背景线程?{2}",Thread.CurrentThread.ManagedThreadId,Thread.CurrentThread.IsThreadPoolThread,Thread.CurrentThread.IsBackground);
//await之前的代码是调用者线程来执行,await之后到下一个await之前的代码由线程池中的同一个线程执行
//但是在在UI程序中,UI线程调用了async方法,则await之后的语句由UI线程来执行,不由线程池中的线程来执行
//TODO 当我们使用await等待一个异步操作时,默认情况下,它会捕获当前线程的同步上下文,等待异步方法执行结束,其后继的代码会被打包到一起,调用SyncContext.Post方法,推送到前面的同步线程上下文中执行
//但是在await后面的语句调用这个方法并传参为false await DoWork().ConfigureAwait(false);可以通知系统不要捕获线程同步上下文,不会被post到await之前捕获的线程上下文中执行,而是直接使用当前完成异步任务的那个线程执行,避免了线程切换
int result = await DoWork();
Console.WriteLine("\n退出TestDoWorkAsync()方法,await语句之后的代码执行线程ID:{0},来自线程池?{1},是背景线程?{2}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread, Thread.CurrentThread.IsBackground);
Console.WriteLine("结果为{0}",result);
} static Task<int> DoWork()
{ return Task.Run(() =>
{
Console.WriteLine("\n使用TPL运行DoWork方法,负责执行的线程ID:{0},来自线程池?{1},是背景线程?{2}",
Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread,
Thread.CurrentThread.IsBackground);
Thread.Sleep();
return ;
});
}

看一下执行结果

可以看到 调用者线程在执行到await这里时会开启一个新的线程去执行await方法,并且立即返回,所以在await DoWork();方法前和TestDoWorkAsync();方法后都是由主线程去执行,而异步方法DoWork()和DoWork之后的语句都是由新线程去执行

看一下调用过程

                              

小结:同一个方法中的代码,以await为边界,被划分为两块或者多块(取决于await语句有多少个),然后,会由线程池中的某个线程来负责执行它们

但是在涉及到UI线程时又会有些变化,这也是让我很难理解的一点

比如在按钮响应事件中调用这个方法

private async void UseAsync()
{
lblInfo.Text = "等待后台程序完成";
btnLanch.Enabled = false;
string result = await SayHelloToAsync("张三"); lblInfo.Text = result;
btnLanch.Enabled = true;
} private Task<string> SayHelloToAsync(string name)
{
return Task.Factory.StartNew(() => SayHelloTo(name));
}

这里在执行完SayHelloToAsync()方法之后,UI控件会得到更新,淫威UseAsync方法的调用是在用户单击按钮引发的,是在UI线程中启动的异步调用,所以在执行完之后,后面的代码会被推送到UI线程中会执行

当我们使用await等待一个异步操作时,默认情况下,它会捕获当前线程的同步上下文,等待异步方法执行结束,其后继的代码会被打包到一起,
调用SyncContext.Post方法,推送到前面的同步线程上下文中执行
根据我的实践,这种情况只会出现在有UI界面的的程序中,在控制台程序中,await异步执行完之后,后面的代码还是会由新线程执行,不会由调用者线程执行。
但是感觉这里有坑,先挖在这,之后再填。(我猜测在控制台程序中,await的线程同步上下文就是新线程而不是调用者线程)

C#5.0 异步编程 Async和Await--理解异步方法与线程之间的关系的更多相关文章

  1. C#异步编程(async and await)及异步方法同步调用

    1.什么是异步? 异步操作通常用于执行完成时间可能较长的任务,如打开大文件.连接远程计算机或查询数据库=异步操作在主应用程序线程以外的线程中执行.应用程序调用方法异步执行某个操作时,应用程序可在异步方 ...

  2. C#5.0 异步编程async/await用法

    微软在发布VS2012的同时推出了C#5.0,其中包含了async和await 代码如下: class Program { private static readonly Stopwatch watc ...

  3. .NET4.5 异步编程 async和await

    msdn介绍:https://msdn.microsoft.com/zh-cn/library/hh191443.aspx 其实很简单,标记了async的方法为异步方法,从方法的左大括号开始同步执行, ...

  4. 异步编程- async和await

    使用目的 避免阻塞主线程 提高程序响应能力 C#中使用 C# 中的 Async 和 Await 关键字是异步编程的核心. 疑惑 The async and await keywords don't c ...

  5. C#中的异步编程Async 和 Await

    谈到C#中的异步编程,离不开Async和Await关键字 谈到异步编程,首先我们就要明白到底什么是异步编程. 平时我们的编程一般都是同步编程,所谓同步编程的意思,和我们平时说的同时做几件事情完全不同. ...

  6. .net异步编程async和await的讨论收获

    微软官方描述: C# 5 引入了一种简便方法,即异步编程.此方法利用了 .NET Framework 4.5 及更高版本..NET Core 和 Windows 运行时中的异步支持. 编译器可执行开发 ...

  7. C#5.0 异步编程 Async和Await--异步方法的规范和注意事项

    要些异步方法要注意一下几点: 异步方法的返回值有三种: 1.没有任何返回值的void 2.返回一个Task任务的Task,可以获得该异步方法的执行状态 3.返回Task<T> 可以获得异步 ...

  8. C#5.0 异步编程 Async和Await--介绍

    C#5.0引入async和await关键字实现方法的异步调用. 直接进入正题. async只是一个标识符,并没有实际的用途,只是用于表明某个方法是异步方法,在方法前面加上async 表示该方法为一个异 ...

  9. C#异步编程----async和await组合的写法

    微软示例: private async void StartButton_Click(object sender, RoutedEventArgs e) { // ExampleMethodAsync ...

随机推荐

  1. SpringMVC学习(10):异常处理

    在项目中如何处理出现的异常,在每个可能出现异常的地方都写代码捕捉异常?这显然是不合理的,当项目越来越大是也是不可维护的.那么如何保证我们处理异常的代码精简且便于维护呢?这就是本篇要讲的内容->异 ...

  2. css中的文本字间距离、行距、overflow

    css字间距.div css字符间距样式实例1.text-indent设置抬头距离css缩进 div设置css样式text-indent : 20px; 缩进了20px 2.letter-spacin ...

  3. ollvm 新增字符串加密功能

    好久没弄ollvm了,可以继续了,今天给ollvm新增了一个pass,用来加密字符串,这个pass是从别的库里面扒出来的. 本文是基于在Windows 上使用VS2017编译出来的ollvm,在这个基 ...

  4. ps学习记录

    基本快捷键: ctrl + 放大 ctrl - 缩小 ctrl 空格键 放大工具 ctrl 0 适合屏幕大小 ctrl 1 显示实际大小 ctrl n 新建画布 ctrl v 移动工具 按住alt键 ...

  5. go语言从例子开始之Example18.struct结构体

    Go 的结构体 是各个字段字段的类型的集合.这在组织数据时非常有用 Example: package main import "fmt" type product struct{ ...

  6. Docker配置阿里云镜像加速pull

    前言:默认Docker拉取镜像是从Docker Hub上拉取,但由于防火墙的原因,导致镜像下载非常慢.为了提高拉取镜像的速度,可以配置阿里镜像或是网易镜像加速,通过使用经验推荐配置阿里镜像. 申请个人 ...

  7. 【Java程序】约瑟夫环

    今天看视频教程无意间看到了一个数3减1的问题,百度之发现叫约瑟夫环问题,于是写了程序,问题大致描述如下: 一群带有编号的孩子手拉手围成一个圈报数,开始的孩子数1,他右边数2,再右边数3,数到n的孩子o ...

  8. DB2数据库常用的函数总结

    CONCAT>>-CONCAT-------(--expression1--,--expression2--)--------------><功能:将两个字符串连接起来,如果两 ...

  9. 如何用Word制作斜线表头?

    如何用Word制作斜线表头?遇到这种问题,你一般是如何操作?本期企业网盘坚果云干货分享与大家分享有关斜线表头的制作方法. 斜线表头分单斜线表头和多斜线表头,下面分情况来了解相关的解决办法. 单斜线表头 ...

  10. 常用Concurrent.util包工具类——高并发

    一 Concurrent.util常用类: 1. CyclicBarrier: 假设有场景:每个线程代表一个跑步运动员,当运动员都准备好后,才一起出发只要有一个人没有准备好,大家都等待. import ...