原文

避免async void

async void异步方法只有一个目的:使得event handler异步可行,也就是说async void只能用于event handler。

async void方法有不同的错误处理机制。当async Task或者async Task<T>方法里面抛出异常时,异常会被捕捉到,并放在Task对象里面。在async void方法里面没有Task对象,因此发生在async void方法里面的异常会直接raised到SynchronizationContext 中。发生在async void方法里面的异常不会被正常捕捉到。

private async void ThrowExceptionAsync()
{
throw new InvalidOperationException();
} public void AsyncVoidExceptions_CannotBeCaughtByCatch()
{
try
{
ThrowExceptionAsync();
}
catch (Exception)
{
// 不会捕捉到这个异常
throw;
}
}

返回Task或者Task<T>的异步方法,可以使用await, Task.WhenAny, Task.WhenAll观察到异步方法是什么时候结束的。async void方法不容易观察到什么时候完成的。async void方法完成后会通知到他们的SynchronizationContext,但是太麻烦了。

Async void方法不利于测试。

死锁

public static class DeadlockDemo
{
private static async Task DelayAsync()
{
await Task.Delay(1000);
} // 在GUI和ASP.NET下这个方法会导致死锁
public static void Test()
{
// Start the delay.
var delayTask = DelayAsync();
// Wait for the delay to complete.
delayTask.Wait();
}
}

死锁的根本原因是await的处理上下文的方式。默认情况下,当一个未完成的Task处于等待时,会捕捉到当前的上下文用于当Task完成时回到该方法。这个上下文就是SynchronizationContext。GUI 和ASP.NET 应用的SynchronizationContext一次只允许一段代码运行。当await完成,它企图在捕捉到的上下文中执行async方法剩下来的代码。但是这个上下文已经有一个线程在跑了,这个线程就是等待这个async方法运行完的方法(在这里就是Test这个同步方法)。这两个方法相互等待对方完成,因此发生了死锁。

控制台应用不会发生死锁。因为它有一个线程池的SynchronizationContext而不是GUI 和ASP.NET 应用一次只允许一段代码运行的SynchronizationContext

解决这个死锁的最好的方案是让Test这个方法也是异步的。控制台方法不能用这个方案,因为Main方法不能是异步的。如果Main方法是异步的,它会在调用的异步方法前返回。

控制台的Main方法是少有的可以调用异步方法的非异步方法。

class Program
{
static void Main()
{
MainAsync().Wait();
} static async Task MainAsync()
{
try
{
// Asynchronous implementation.
await Task.Delay(1000);
}
catch (Exception ex)
{
// Handle exceptions.
}
}
}

Configure Context

回到上面死锁的问题,当一个未完成的Task处于await的时候,默认会捕捉到上下文,捕捉到的这个上下文用于重回异步方法。context的行为会导致另外一个问题---性能问题。As asynchronous GUI applications grow larger, you might find many small parts of async methods all using the GUI thread as their context. This can cause sluggishness as responsiveness suffers from “thousands of paper cuts.”

减少这个导致的性能问题,可以通过ConfigureAwait来实现。

async Task MyMethodAsync()
{
// 代码在最初的上下文中运行
await Task.Delay(1000); // 代码在最初的上下文中运行
await Task.Delay(1000).ConfigureAwait(continueOnCapturedContext: false); // 代码不在最初的上下文中运行了
// 现在的上下文在线程池
}

除了性能外,ConfigureAwait的另外一个重要的作用是可以避免死锁。

[译]Async/Await - Best Practices in Asynchronous Programming的更多相关文章

  1. Async/Await - Best Practices in Asynchronous Programming z

    These days there’s a wealth of information about the new async and await support in the Microsoft .N ...

  2. Async/Await - Best Practices in Asynchronous Programming

    https://msdn.microsoft.com/en-us/magazine/jj991977.aspx Figure 1 Summary of Asynchronous Programming ...

  3. [译]async/await中使用阻塞式代码导致死锁 百万数据排序:优化的选择排序(堆排序)

    [译]async/await中使用阻塞式代码导致死锁 这篇博文主要是讲解在async/await中使用阻塞式代码导致死锁的问题,以及如何避免出现这种死锁.内容主要是从作者Stephen Cleary的 ...

  4. [译]async/await中使用阻塞式代码导致死锁

    原文:[译]async/await中使用阻塞式代码导致死锁 这篇博文主要是讲解在async/await中使用阻塞式代码导致死锁的问题,以及如何避免出现这种死锁.内容主要是从作者Stephen Clea ...

  5. [译]async/await中阻塞死锁

    这篇博文主要是讲解在async/await中使用阻塞式代码导致死锁的问题,以及如何避免出现这种死锁.内容主要是从作者Stephen Cleary的两篇博文中翻译过来. 原文1:Don'tBlock o ...

  6. Best Practices in Asynchronous Programming

    http://blog.stephencleary.com/ http://blogs.msdn.com/b/pfxteam/

  7. 将 async/await 异步代码转换为安全的不会死锁的同步代码

    在 async/await 异步模型(即 TAP Task-based Asynchronous Pattern)出现以前,有大量的同步代码存在于代码库中,以至于这些代码全部迁移到 async/awa ...

  8. Async/Await FAQ

    From time to time, I receive questions from developers which highlight either a need for more inform ...

  9. 异步模式:Callbacks, Promises & Async/Await

    [译]异步JavaScript的演变史:从回调到Promises再到Async/Await https://www.i-programmer.info/programming/theory/8864- ...

随机推荐

  1. 利用ZYNQ SOC快速打开算法验证通路(6)——LWIP实现千兆TCP/IP网络传输

    一.前言 之前ZYNQ与PC之间的网络连接依赖于外接硬件协议栈芯片,虽然C驱动非常简单,但网络带宽受限.现采用LWIP+PS端MAC控制器+PHY芯片的通用架构.关于LWIP库,已经有很多现成的资料和 ...

  2. sql 视图学习

    视图语句 在 SQL 中,视图是基于 SQL 语句的结果集的可视化的表. 视图包含行和列,就像一个真实的表.视图中的字段就是来自一个或多个数据库中的真实的表中的字段. 您可以向视图添加 SQL 函数. ...

  3. Docker + Sonarqube 环境搭建

    Sonar概述 Sonar 是一个用于代码质量管理的开放平台.通过插件机制,Sonar 可以集成不同的测试工具,代码分析工具,以及持续集成工具. 一,postgres 数据库下载 docker pul ...

  4. java有关 String char 常见问题 编辑中

    1 输入输出有关 Scanner 的next()方法 返回值是String 所以尝试获得char时 应该用input.next().charAt[0] 2 空值 String 中null是指 对象引用 ...

  5. 应用wavesurfer.js绘制音频波形图小白极速上手总结

    一.简介 1.1  引   人生中第一份工作公司有语音识别业务,需要做一个web网页来整合语音引擎的标注结果和错误率等参数,并提供人工比对的语音标注功能(功能类似于TranscriberAG等),(博 ...

  6. ERROR:"org.apache.zookeeper.KeeperException$NoAuthException: KeeperErrorCode = NoAuth for /config/topics/test" when creating or deleting Kafka operations authorized through the Ranger policies

    PROBLEM DESCRIPTION When creating or deleting topics in Kafka, they cannot be authorized through the ...

  7. 用Redis管理Session

    maven <dependency> <groupId>redis.clients</groupId> <artifactId>jedis</ar ...

  8. 类ArrayList

    什么是ArrayList类 Java提供了一个容器 java.util.ArrayList 集合类,他是大小可变的数组的实现,存储在内的数据称为元素.此类提供一些方法来操作内部存储的元素. Array ...

  9. python中使用redis发布订阅者模型

    redis发布订阅者模型: Redis提供了发布订阅功能,可以用于消息的传输,Redis的发布订阅机制包括三个部分,发布者,订阅者和Channel.发布者和订阅者都是Redis客户端,Channel则 ...

  10. Spring Cloud Netflix vs Spring Cloud Alibaba

    Spring Cloud Netflixhttps://spring.io/projects/spring-cloud-netflix spring-cloud-alibaba/README-zh.m ...