为什么不要使用 async void的原因分析

网友投稿 210 2023-07-05

为什么不要使用 async void的原因分析

问题

在使用 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 EnqueueJob()

{

await _backgroundJobManager.EnqueueAsync("测试文本。");

return "执行完成。";

}

}

public class BG : BackgroundJob, 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 替换掉之前的 BackgroundJob ,只需要实现它的 Task ExecuteAsync(TArgs args) 方法即可。

public class BGAsync : AsyncBackgroundJob,ITransientDependency

{

private readonly TestAppService _testAppService;

public BGAsync(TestAppService testAppService)

{

_testAppService = testAppService;

}

protected override async Task ExecuteAsync(string args)

{

await _testAppService.GetInvalidOperationExcepivDPRxrGtion();

}

}

总结

以上所述是给大家介绍的为什么不要使用 async void的原因分析,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,会及时回复大家的!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:springboot多环境(dev、test、prod)配置详解
下一篇:详解Java的TCP/IP编程学习
相关文章

 发表评论

暂时没有评论,来抢沙发吧~