為什么不要使用 async void的原因分析
問題
在使用 Abp 框架的后臺作業(yè)時,當(dāng)后臺作業(yè)拋出異常,會導(dǎo)致整個程序崩潰。在 Abp 框架的底層執(zhí)行后臺作業(yè)的時候,有 try/catch 語句塊用來捕獲后臺任務(wù)執(zhí)行時的異常,但是在這里沒有生效。
原始代碼如下:
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 "執(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)用接口時的效果:

原因
出現(xiàn)這種情況是因為任何異步方法返回 void 時,拋出的異常都會在 async void 方法啟動時,處于激活狀態(tài)的同步上下文 (SynchronizationContext) 觸發(fā),我們的所有 Task 都是放在線程池執(zhí)行的。
所以在上述樣例當(dāng)中,此時 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 來監(jiān)聽異常,不過你是沒辦法從異常狀態(tài)恢復(fù)的。
解決
可以使用 AsyncBackgroundJob<TArgs> 替換掉之前的 BackgroundJob<TArgs> ,只需要實現(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的原因分析,希望對大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會及時回復(fù)大家的!
相關(guān)文章
解決SpringMvc中普通類注入Service為null的問題
這篇文章主要介紹了解決SpringMvc中普通類注入Service為null的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
Java實現(xiàn)為Word每一頁設(shè)置不同圖片水印的效果
Word中設(shè)置水印時,可加載圖片設(shè)置為水印效果,但通常添加水印效果時,會對所有頁面都設(shè)置成統(tǒng)一效果。所以本文為大家介紹了一個方法,可以實現(xiàn)對每一頁或者某個頁面設(shè)置不同的水印效果,需要的可以參考一下2022-02-02
spring實現(xiàn)靜態(tài)注入(類或者屬性)操作示例
這篇文章主要為大家介紹了spring實現(xiàn)靜態(tài)注入(類或者屬性)操作示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07
NameNode?重啟恢復(fù)數(shù)據(jù)的流程詳解
這篇文章主要為大家介紹了NameNode?重啟恢復(fù)數(shù)據(jù)的流程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02
Java使用JSON實現(xiàn)處理中文亂碼和Date格式
這篇文章主要為大家詳細介紹了Java如何在項目中使用JSON實現(xiàn)處理中文亂碼和Date格式的功能,文中的示例代碼講解詳細,需要的小伙伴可以參考一下2023-06-06

