《C#并发编程经典实例》学习笔记—异步编程关键字 Async和Await
C# 5.0 推出async和await,最早是.NET Framework 4.5引入,可以在Visual Studio 2012使用。在此之前的异步编程实现难度较高,async使异步编程的实现变得简便。
各平台对async的支持情况
| 平台 | async |
|---|---|
| .NET 4.5及以上 | ✓ |
| .NET 4.0 | NuGet |
| Mono iOS/Droid | ✓ |
| Windows Store | ✓ |
| Windows Phone Apps 8.1 | ✓ |
| Windows Phone SL 8.0 | ✓ |
| Windows Phone SL 7.1 | NuGet |
| Silverlight 5 | NuGet |
在不支持的平台,安装NuGet包 Microsoft.Bcl.Async
使用
async修饰符可将方法、lambda 表达式或匿名方法指定为异步。
async 对方法做了什么处理
从使用async修饰符修饰的方法的IL代码可以得出一个结论:
- 在Debug下,针对async方法,生成的是一个class状态机
- 在Release下,针对async方法,生成的是一个struct状态机
举例:
C#代码如下
using System.Threading.Tasks;
namespace ConsoleApp3
{
public class Test
{
public async Task TestAsync()
{
await GetAsync();
}
public async Task GetAsync()
{
await Task.Delay(1);
}
}
}
以TestAsync方法为准
Release下 初始化状态机V_0 ,类型是值类型Struct(valuetype),类型名称为<TestAsync>d__0
.locals init (
[0] valuetype ConsoleApp3.Test/'<TestAsync>d__0' V_0,
[1] valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder V_1
)
<TestAsync>d__0 继承值类型[mscorlib]System.ValueType
.class nested private sealed auto ansi beforefieldinit
'<TestAsync>d__0'
extends [mscorlib]System.ValueType
implements [mscorlib]System.Runtime.CompilerServices.IAsyncStateMachine
Debug 下 初始化状态机V_0 ,类型是引用类型Class(class) ,类型名称为<TestAsync>d__0
.locals init (
[0] class ConsoleApp3.Test/'<TestAsync>d__0' V_0,
[1] valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder V_1
)
<TestAsync>d__0 继承引用类型[mscorlib]System.Object
.class nested private sealed auto ansi beforefieldinit
'<TestAsync>d__0'
extends [mscorlib]System.Object
implements [mscorlib]System.Runtime.CompilerServices.IAsyncStateMachine
异步方法的定义和注意事项
使用
async关键字定义的异步方法简称为“异步方法”。
注意事项:
- 如果
async关键字修改的方法不包含await表达式或语句,则该方法将同步执行。 编译器警告将通知你不包含await语句的任何异步方法,因为该情况可能表示存在错误。 请参阅编译器警告(等级 1)CS4014。 async关键字是上下文关键字,原因在于只有当它修饰方法、lambda 表达式或匿名方法时,它才是关键字。 在所有其他上下文中,都会将其解释为标识符。- 不要用
void作为async方法的返回类型!async方法可以返回void,但是这仅限于编写事件处理程序。一个普通的async方法如果没有返回值,要返回Task,而不是void。 - 一定要避免使用
Task.Wait或Task<T>.Result方法,因为它们会导致死锁。如果使用了async,最好就一直使用它。 - 异步方法的参数不能使用
out,ref。out或ref返回的数据应借用Task<TResult>返回,可以使用元组或自定义数据结构。
异步方法的特征
- 方法签名包含
async修饰符。 - 按照约定,异步方法的名称以“Async”后缀结尾。
- 返回类型为下列类型之一:
- 如果你的方法有操作数为
TResult类型的返回语句,则为Task<TResult>。 - 如果你的方法没有返回语句或具有没有操作数的返回语句,则为
Task。 void:如果要编写异步事件处理程序。- 包含
GetAwaiter方法的其他任何类型(自 C# 7.0 起)。
- 如果你的方法有操作数为
- 方法通常包含至少一个 await 表达式,该表达式标记一个点,在该点上,直到等待的异步操作完成方法才能继续。 同时,将方法挂起,并且控制返回到方法的调用方。
关于async和await具体的执行流程,方法何时挂起和释放,请参考异步程序中的控制流 (C#)
异步返回类型
上面提到 void 作为返回结果,适用于事件处理程序。
举例:
using System;
using System.Threading.Tasks;
namespace ConsoleApp3
{
public class TestVoidAsync
{
private event EventHandler<EventArgs> DoTest;
public TestVoidAsync()
{
DoTest += DoTestEvent;
}
private static async void DoTestEvent(object sender, EventArgs e)
{
await Task.Delay(1000);
}
protected virtual void OnDoTest()
{
DoTest?.Invoke(this, EventArgs.Empty);
}
}
}
void 作为返回结果存在一个弊端:无法捕获异常。
返回 void 的异步方法的调用方无法捕获从该方法引发的异常,且此类未经处理的异常可能会导致应用程序故障。 如果返回
Task或Task<TResult>的异步方法中出现异常,此异常将存储于返回的任务中,并在等待该任务时再次引发。
通用的异步返回类型:
从 C# 7.0 开始,异步方法可返回任何具有可访问的 GetAwaiter 方法的类型。
ValueTask<TResult>:
Task 和 Task<TResult> 是引用类型,因此,性能关键路径中的内存分配会对性能产生负面影响,尤其当分配出现在紧凑循环中时。 支持通用返回类型意味着可返回轻量值类型(而不是引用类型),从而避免额外的内存分配。
使用ValueTask<TResult>,需要添加NuGet包 System.Threading.Tasks.Extensions。
ValueTask<TResult> 是struct值类型,Task 和 Task<TResult> 是class引用类型
异步操作的生命周期
Task 类提供了异步操作的生命周期,且该周期由 TaskStatus 枚举表示。
| 状态 | 执行顺序 | 备注 |
|---|---|---|
| Created | 0 | 该任务已初始化,但尚未安排。 |
| WaitingForActivation | 1 | 该任务正在等待被.NET Framework infrastructure 内部激活和调度。 |
| WaitingToRun | 2 | 该任务已安排执行但尚未开始执行。 |
| Running | 3 | 任务正在运行但尚未完成。 |
| WaitingForChildrenToComplete | 4 | 任务已完成执行,并隐式等待附加的子任务完成。 |
| RanToCompletion | 5 | 任务已成功完成执行。 |
| Canceled | 6 | 引发 OperationCanceledException 异常,或者在任务开始执行之前取消 |
| Faulted | 7 | 由于未处理的异常,任务已完成。 |
Canceled 和 Faulted状态都会因为任务异常导致转换为该状态。二者的区别如下:
如果标记的 IsCancellationRequested 属性返回 false,或者异常的标记与任务的标记不匹配,则会将 OperationCanceledException 按照普通的异常来处理,从而导致任务转换为 Faulted 状态。 另外还要注意,其他异常的存在将也会导致任务转换为 Faulted 状态。 您可以在 Status 属性中获取已完成任务的状态。
参考文章:
《C#并发编程经典实例》学习笔记—异步编程关键字 Async和Await的更多相关文章
- 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换
[源码下载] 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换 作者:webabcd 介绍 ...
- 《C#并发编程经典实例》笔记
1.前言 2.开宗明义 3.开发原则和要点 (1)并发编程概述 (2)异步编程基础 (3)并行开发的基础 (4)测试技巧 (5)集合 (6)函数式OOP (7)同步 1.前言 最近趁着项目的一段平稳期 ...
- nodejs学习笔记 —— 异步编程解决方案
在js或者node编程中,由于异步的频繁和广度使用,使得回调和嵌套的深度导致编程的体验遇到一些挑战,如果写出优雅和好看的代码,本文主要针对异步编程的主流方案做一些总结 1.事件发布/订阅模式 事件监听 ...
- 09-Node.js学习笔记-异步编程
同步API,异步API 同步API:只有当前API执行完成后,才能继续执行下一个API console.log('before'); console.log('after'); 异步API:当前API ...
- 多线程之异步编程: 经典和最新的异步编程模型,async与await
经典的异步编程模型(IAsyncResult) 最新的异步编程模型(async 和 await) 将 IAsyncInfo 转换成 Task 将 Task 转换成 IAsyncInfo 示例1.使用经 ...
- 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换
经典的异步编程模型(IAsyncResult) 最新的异步编程模型(async 和 await) 将 IAsyncInfo 转换成 Task 将 Task 转换成 IAsyncInfo 示例1.使用经 ...
- 《C#并发编程经典实例》学习笔记—2.7 避免上下文延续
避免上下文延续 在默认情况下,一个 async 方法在被 await 调用后恢复运行时,会在原来的上下文中运行. 为了避免在上下文中恢复运行,可让 await 调用 ConfigureAwait 方法 ...
- 《C#并发编程经典实例》学习笔记—3.1 数据的并行处理
问题 有一批数据,需要对每个元素进行相同的操作.该操作是计算密集型的,需要耗费一定的时间. 解决方案 常见的操作可以粗略分为 计算密集型操作 和 IO密集型操作.计算密集型操作主要是依赖于CPU计算, ...
- 《C#并发编程经典实例》学习笔记—2.3 报告任务
问题 异步操作时,需要展示该操作的进度 解决方案 IProgress<T> Interface和Progress<T> Class 插一段话:读<C#并发编程经典实例&g ...
随机推荐
- Android开发者的Anko使用指南(二)之Dialogs
在项目中使用Anko Dialogs dependencies { compile "org.jetbrains.anko:anko-commons:$anko_version" ...
- sqlserver 多行转一行
sql 例子: SELECT STUFF((SELECT ',' + CONVERT(VARCHAR, b.SCsinfoSourceId) FROM PZDataCsinfo b WHERE b.D ...
- wxPyhon 的控件(摘抄)
一.静态文本控件 wx.StaticText(parent, id, label, pos=wx.DefaultPosition, size=wx.DefaultSize, style=0, n ...
- linux 性能优化
linux的性能优化: 1.CPU,MEM 2.DISK--RAID 3.网络相关的外设,网卡 linux系统性能分析: top:linux系统的负载,CPU,MEM,SWAP,占用CPU和内存比较的 ...
- Redis安装和实际应用
上次介绍了Redis的来龙去脉以及相关一些情况,点击回顾<深入浅出Redis>,接下来我们再讲讲Redis的安装和实际应用. 一.Redis的安装 下载安装包,redis-3.2.9.ta ...
- Java实现链表的常见操作算法
链表分为单链表,双向链表和循环链表,是一种链式存储结构,由一个个结点链式构成,结点包含数据域和指针域,其中单链表是只有一个指向后驱结点的指针,双向链表除头结点和尾结点外,每个结点都有一个前驱指针和一个 ...
- Java面试集合(六)
1. abstract抽象 什么是abstract,中文为抽象,从具体事物抽出,概括它们共同的方面,本质属性与关系等,称为抽象.看不见,摸不着的东西叫做抽象,抽象是人们对世界万物的感觉,用特定的图像表 ...
- Oracle merge合并更新函数
本博客介绍一下Oracle merge合并函数,业务场景:新增数据的时候要先查询数据库是否已经有改数据,有数据就更新数据,没数据才新增数据,这是很常见的业务场景,如果是用Oracle数据库的话,其实直 ...
- Python - 使用Setuptools进行程序打包
1- Setuptools简介 通过Setuptools可以更方便的创建和发布Python包,特别是那些对其它包具有依赖性的状况: Python打包用户指南(Python Packaging User ...
- python基础-变量运算符(3)
一.注释 注释就是对代码的解释和说明.目的是为了让别人和自己很容易看懂.为了让别人一看就知道这段代码是做什么用的.正确的程序注释一般包括序言性注释和功能性注释.序言性注释的主要内容包括模块的接口.数据 ...