为什么不要使用 async void的原因分析
作者:myzony 时间:2023-11-24 21:10:27
问题
在使用 Abp 框架的后台作业时,当后台作业抛出异常,会导致整个程序崩溃。在 Abp 框架的底层执行后台作业的时候,有 try/catch 语句块用来捕获后台任务执行时的异常,但是在这里没有生效。
原始代码如下:
public class TestAppService : ITestAppService
{
private readonly IBackgroundJobManager _backgroundJobManager;
public TestAppService(IBackgroundJobManager backgroundJobManager)
{
_backgroundJobManager = backgroundJobManager;
}
public Task GetInvalidOperationException()
{
throw new InvalidOperationException("模拟无效操作异常。");
}
public async Task<string> EnqueueJob()
{
await _backgroundJobManager.EnqueueAsync<BG, string>("测试文本。");
return "执行完成。";
}
}
public class BG : BackgroundJob<string>, ITransientDependency
{
private readonly TestAppService _testAppService;
public BG(TestAppService testAppService)
{
_testAppService = testAppService;
}
public override async void Execute(string args)
{
await _testAppService.GetInvalidOperationException();
}
}
调用接口时的效果:
原因
出现这种情况是因为任何异步方法返回 void 时,抛出的异常都会在 async void 方法启动时,处于激活状态的同步上下文 (SynchronizationContext)
触发,我们的所有 Task 都是放在线程池执行的。
所以在上述样例当中,此时 AsyncVoidMethodBuilder.Create()
使用的同步上下文为 null ,这个时候 ThreadPool 就不会捕获异常给原有线程处理,而是直接抛出。
线程池在底层使用 AsyncVoidMethodBuilder.Craete()
所拿到的同步上下文,所捕获异常的代码如下:
internal static void ThrowAsync(Exception exception, SynchronizationContext targetContext)
{
var edi = ExceptionDispatchInfo.Capture(exception);
// 同步上下文是空的,则不会做处理。
if (targetContext != null)
{
try
{
targetContext.Post(state => ((ExceptionDispatchInfo)state).Throw(), edi);
return;
}
catch (Exception postException)
{
edi = ExceptionDispatchInfo.Capture(new AggregateException(exception, postException));
}
}
}
虽然你可以通过挂载 AppDoamin.Current.UnhandledException
来监听异常,不过你是没办法从异常状态恢复的。
解决
可以使用 AsyncBackgroundJob<TArgs>
替换掉之前的 BackgroundJob<TArgs>
,只需要实现它的 Task ExecuteAsync(TArgs args)
方法即可。
public class BGAsync : AsyncBackgroundJob<string>,ITransientDependency
{
private readonly TestAppService _testAppService;
public BGAsync(TestAppService testAppService)
{
_testAppService = testAppService;
}
protected override async Task ExecuteAsync(string args)
{
await _testAppService.GetInvalidOperationException();
}
}
总结
以上所述是小编给大家介绍的为什么不要使用 async void的原因分析,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!
来源:https://www.cnblogs.com/myzony/p/10647460.html
标签:async,void
0
投稿
猜你喜欢
深入理解Kotlin的泛型系统
2023-09-09 00:47:30
SpringData如何通过@Query注解支持JPA语句和原生SQL语句
2022-08-26 22:07:29
Java中多线程下载图片并压缩能提高效率吗
2023-08-06 07:40:10
android 仿微信demo——登录功能实现(服务端)
2023-10-04 13:40:55
你所不知道的Spring自动注入详解
2021-09-04 19:30:08
用java等语言仿360首页拼音输入全模糊搜索和自动换肤
2022-12-28 18:32:01
android LinearLayout和RelativeLayout组合实现精确布局方法介绍
2021-06-17 12:33:33
java实现学生成绩档案管理系统
2023-06-29 22:48:43
java使用软引用实现缓存机制示例
2021-08-26 18:06:12
Spring Cloud 动态刷新配置信息教程详解
2023-12-02 04:48:16
C#适配器模式的使用
2022-02-05 15:47:21
mybatis教程之resultmap_动力节点Java学院整理
2022-09-05 22:38:37
HTTP中get和post的区别详解
2023-04-19 11:42:18
Android中多个ContentProvider的初始化顺序详解
2021-11-06 03:52:36
Spring mvc Json处理实现流程代码实例
2023-07-14 21:31:06
基于Android本地代码生成器详解
2022-09-15 02:52:26
JAVA基于SnakeYAML实现解析与序列化YAML
2023-11-26 06:14:44
java实现简单登录界面的实战过程
2022-02-07 20:19:51
Java实现的不同图片居中剪裁生成同一尺寸缩略图功能示例
2023-08-23 14:53:15
C# 7.2中结构体性能问题的解决方案
2022-08-12 23:04:26