為什么不要使用 async void的原因分析
問(wèn)題
在使用 Abp 框架的后臺(tái)作業(yè)時(shí),當(dāng)后臺(tái)作業(yè)拋出異常,會(huì)導(dǎo)致整個(gè)程序崩潰。在 Abp 框架的底層執(zhí)行后臺(tái)作業(yè)的時(shí)候,有 try/catch 語(yǔ)句塊用來(lái)捕獲后臺(tái)任務(wù)執(zhí)行時(shí)的異常,但是在這里沒(méi)有生效。
原始代碼如下:
public class TestAppService : ITestAppService
{
private readonly IBackgroundJobManager _backgroundJobManager;
public TestAppService(IBackgroundJobManager backgroundJobManager)
{
_backgroundJobManager = backgroundJobManager;
}
public Task GetInvalidOperationException()
{
throw new InvalidOperationException("模擬無(wú)效操作異常。");
}
public async Task<string> EnqueueJob()
{
await _backgroundJobManager.EnqueueAsync<BG, string>("測(cè)試文本。");
return "執(zhí)行完成。";
}
}
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();
}
}
調(diào)用接口時(shí)的效果:

原因
出現(xiàn)這種情況是因?yàn)槿魏萎惒椒椒ǚ祷?void 時(shí),拋出的異常都會(huì)在 async void 方法啟動(dòng)時(shí),處于激活狀態(tài)的同步上下文 (SynchronizationContext) 觸發(fā),我們的所有 Task 都是放在線程池執(zhí)行的。
所以在上述樣例當(dāng)中,此時(shí) AsyncVoidMethodBuilder.Create() 使用的同步上下文為 null ,這個(gè)時(shí)候 ThreadPool 就不會(huì)捕獲異常給原有線程處理,而是直接拋出。
線程池在底層使用 AsyncVoidMethodBuilder.Craete() 所拿到的同步上下文,所捕獲異常的代碼如下:
internal static void ThrowAsync(Exception exception, SynchronizationContext targetContext)
{
var edi = ExceptionDispatchInfo.Capture(exception);
// 同步上下文是空的,則不會(huì)做處理。
if (targetContext != null)
{
try
{
targetContext.Post(state => ((ExceptionDispatchInfo)state).Throw(), edi);
return;
}
catch (Exception postException)
{
edi = ExceptionDispatchInfo.Capture(new AggregateException(exception, postException));
}
}
}
雖然你可以通過(guò)掛載 AppDoamin.Current.UnhandledException 來(lái)監(jiān)聽(tīng)異常,不過(guò)你是沒(méi)辦法從異常狀態(tài)恢復(fù)的。
解決
可以使用 AsyncBackgroundJob<TArgs> 替換掉之前的 BackgroundJob<TArgs> ,只需要實(shí)現(xiàn)它的 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();
}
}
總結(jié)
以上所述是小編給大家介紹的為什么不要使用 async void的原因分析,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)歡迎給我留言,小編會(huì)及時(shí)回復(fù)大家的!
相關(guān)文章
Maven創(chuàng)建項(xiàng)目過(guò)慢的4種解決辦法
最近經(jīng)常會(huì)遇到一個(gè)困擾,那就是用idea創(chuàng)建maven項(xiàng)目時(shí),速度很慢,本文就來(lái)介紹一下Maven創(chuàng)建項(xiàng)目過(guò)慢的4種解決辦法,感興趣的可以了解一下2021-12-12
Mybatis逆向工程實(shí)現(xiàn)連接MySQL數(shù)據(jù)庫(kù)
本文主要介紹了Mybatis逆向工程實(shí)現(xiàn)連接MySQL數(shù)據(jù)庫(kù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06
Spring boot2X負(fù)載均衡和反向代理實(shí)現(xiàn)過(guò)程解析
這篇文章主要介紹了Spring boot2X負(fù)載均衡和反向代理實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
java驗(yàn)證用戶是否已經(jīng)登錄 java實(shí)現(xiàn)自動(dòng)登錄
這篇文章主要介紹了java驗(yàn)證用戶是否已經(jīng)登錄,java實(shí)現(xiàn)自動(dòng)登錄,感興趣的小伙伴們可以參考一下2016-04-04
SpringCloud Ribbon負(fù)載均衡工具使用
Ribbon是Netflix的組件之一,負(fù)責(zé)注冊(cè)中心的負(fù)載均衡,有助于控制HTTP和TCP客戶端行為。Spring?Cloud?Netflix?Ribbon一般配合Ribbon進(jìn)行使用,利用在Eureka中讀取的服務(wù)信息,在調(diào)用服務(wù)節(jié)點(diǎn)時(shí)合理進(jìn)行負(fù)載2023-02-02
MyBatis Mapper中 @Select注解調(diào)用靜態(tài)常量的問(wèn)題分析
在Java編碼中,我們通常會(huì)把這些數(shù)字或者字符串定義在常量類(lèi)或者接口中,可以直接在mapper中也可以使用這些常量就比較好,這篇文章主要介紹了MyBatis Mapper中 @Select注解調(diào)用靜態(tài)常量,需要的朋友可以參考下2023-06-06
Java連接FTP服務(wù)器并使用ftp連接池進(jìn)行文件操作指南
使用FTP最主要的功能是對(duì)文件進(jìn)行管理,下面這篇文章主要給大家介紹了關(guān)于Java連接FTP服務(wù)器并使用ftp連接池進(jìn)行文件操作的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-08-08

