C# 异步 async await
要点
- 完成对await表达式的计算之后,若await表达式后面还有代码(作用域同函数)则由执行await表达式的线程(不是调用方所在的线程)继续执行这些代码
实例证明,await 表达式之后的代码,由执行await表达式的线程,即Task的同一线程,而非调用方法所在线程。 - Task.Result ,采用 Result 属性获取结果使用的是调用方法的线程,会阻塞调用线程。
正文
C#5.0 新增了async和await关键字,使用这两个关键字可以大大简化异步编程
使用 async 关键字可将方法、lambda 表达式或匿名方法标记为异步,即,方法中应该包含一个或多个await表达式,但async关键字本身不会创建异步操作。
这里需要注意一点,若使用async关键字标记的方法中没有使用await关键字(编译器会给出警告但不报错),那么该方法将会以同步方式执行。
定义异步方法的要求
- 使用async关键字来修饰方法
- 在异步方法中使用await关键字(不使用编译器会给出警告但不报错),否则异步方法会以同步方式执行
- 尽量不使用void作为返回类型,若希望异步方法返回void类型,请使用Task 异步方法名称以Async结尾
- 异步方法中不能声明使用ref或out关键字修饰的变量
await和返回类型
- Task
如果在异步方法返回时使用了await关键字,且匿名方法的返回类型是Task,那么我们得到的返回类型是T。若未使用await关键字,则返回类型是Task。未使用await,调用GetStringAsync方法时result是Task类型。
比较如下:分别返回 和 Task
await
在遇到awiat关键字之前,程序是按照代码顺序自上而下以同步方式执行
在遇到await关键字之后,系统做了以下工作:
- List item
- 异步方法将被挂起
- 将控制权返回给调用者
- 使用线程池中的线程(而非额外创建新的线程)来计算await表达式的结果,所以await不会造成程序的阻塞
- 完成对await表达式的计算之后,若await表达式后面还有代码(作用域同函数)则由执行await表达式的线程(不是调用方所在的线程)继续执行这些代码
使用一段代码来进行验证:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Net.Http