0

0

c# Stephen Cleary 关于异步编程的建议 c# async expert

煙雲

煙雲

发布时间:2025-12-29 17:05:02

|

827人浏览过

|

来源于php中文网

原创

Stephen Cleary 强调异步编程的核心是避免错误:禁用 async void(仅限事件处理器),库中默认使用 ConfigureAwait(false),测试时用 async Task 而非 .Result,I/O 用原生异步 API,CPU 密集型操作勿伪异步。

c# stephen cleary 关于异步编程的建议 c# async expert

Stephen Cleary 是 C# 异步编程领域公认的实战派专家,他不是只讲理论的学者,而是长期在真实项目中踩坑、填坑、提炼模式的工程师。他的核心建议不是“怎么写 async”,而是“怎么不写错 async”——尤其在跨线程上下文、测试、UI 集成和库设计等容易静默崩溃的场景。

避免 async void,除非你真在写事件处理程序

这是 Cleary 反复强调的“第一戒律”。async void 方法无法被 await,异常会直接抛到 SynchronizationContext(比如 UI 线程),导致应用崩溃且难以捕获;它也不支持超时、取消或组合(如 Task.WhenAll)。

  • ✅ 正确:事件处理器(如 button_Click)可 async void,因为事件签名强制返回 void
  • ❌ 错误:把业务逻辑方法写成 async void DoWork() —— 改为 async Task DoWorkAsync()
  • ⚠️ 注意:Console.Main 在 .NET 6+ 允许 async Task Main(),但旧版或类库中仍需避免 void

ConfigureAwait(false) 除非你明确需要上下文

默认情况下,await 会尝试“回到原上下文”(如 UI 线程或 ASP.NET 请求上下文)。这在库代码中是危险的:它可能引发死锁(尤其调用 .Result.Wait() 时),也拖慢性能。

  • ✅ 库/工具方法中:所有 await 后加 .ConfigureAwait(false),例如 await stream.ReadAsync(buffer).ConfigureAwait(false)
  • ❌ UI 层或 ASP.NET Controller 中盲目加:可能破坏数据绑定或 HttpContext 访问
  • ? 小技巧:用 Roslyn 分析器(如 ConfigureAwaitChecker)自动检测遗漏点

测试异步方法时,别用 .Result.Wait()

Cleary 明确指出:“单元测试里出现 .Result,基本等于埋雷”。它会阻塞当前线程,在 xUnit/NUnit 的同步测试上下文中极易触发死锁(尤其涉及 SynchronizationContext 时)。

  • ✅ 正确:测试方法本身标记为 async Task,并 await 被测方法
  • ❌ 错误:
    [Fact]
    public void TestGetData() {
        var result = GetDataAsync().Result; // ⚠️ 死锁高发区
        Assert.Equal("OK", result);
    }
  • ? 补充:xUnit 支持 async Task 测试方法;NUnit 需 ≥ 3.0 并启用 AsyncTest 特性

别让 CPU 密集型操作假装“异步”

异步 ≠ 并行。Cleary 强调:await Task.Run(() => ComputeHeavy()) 不是真正的异步 I/O,只是把同步工作扔进线程池——它解决的是 UI 响应问题,但会增加调度开销,且不能缩放。

  • ✅ I/O 绑定操作(HTTP、DB、文件):用原生异步 API(HttpClient.GetAsyncFileStream.ReadAsync
  • ✅ CPU 绑定操作:用 Parallel.ForEachPLINQ,或明确用 Task.Run 并注明“此为线程池卸载,非异步”
  • ❌ 混淆:async Task CalculateAsync() { await Task.Run(() => ExpensiveMath()); } —— 方法名带 Async 却无真正异步语义,误导调用方

真正难的从来不是“怎么让代码跑起来”,而是“怎么让它在并发、中断、失败、升级后依然可靠”。Cleary 的建议之所以被广泛采纳,是因为他始终站在调用链下游、测试覆盖率、部署稳定性这些真实战场说话——比如一个 ConfigureAwait(false) 的缺失,可能在压测时才暴露为 5% 的随机超时,而没人能复现。

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

37

2025.12.04

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

172

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

92

2025.11.27

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

172

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

92

2025.11.27

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

466

2023.08.10

console接口是干嘛的
console接口是干嘛的

console接口是一种用于在计算机命令行或浏览器开发工具中输出信息的工具,提供了一种简单的方式来记录和查看应用程序的输出结果和调试信息。本专题为大家提供console接口相关的各种文章、以及下载和课程。

409

2023.08.08

console.log是什么
console.log是什么

console.log 是 javascript 函数,用于在浏览器控制台中输出信息,便于调试和故障排除。想了解更多console.log的相关内容,可以阅读本专题下面的文章。

475

2024.05.29

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

121

2025.12.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 2.5万人学习

SciPy 教程
SciPy 教程

共10课时 | 0.9万人学习

Sass 教程
Sass 教程

共14课时 | 0.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号