為什么不要使用 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 都是放在線(xiàn)程池執(zhí)行的。
所以在上述樣例當(dāng)中,此時(shí) AsyncVoidMethodBuilder.Create()
使用的同步上下文為 null ,這個(gè)時(shí)候 ThreadPool 就不會(huì)捕獲異常給原有線(xiàn)程處理,而是直接拋出。
線(xiàn)程池在底層使用 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)文章
解決SpringMvc中普通類(lèi)注入Service為null的問(wèn)題
這篇文章主要介紹了解決SpringMvc中普通類(lèi)注入Service為null的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07Java實(shí)現(xiàn)為Word每一頁(yè)設(shè)置不同圖片水印的效果
Word中設(shè)置水印時(shí),可加載圖片設(shè)置為水印效果,但通常添加水印效果時(shí),會(huì)對(duì)所有頁(yè)面都設(shè)置成統(tǒng)一效果。所以本文為大家介紹了一個(gè)方法,可以實(shí)現(xiàn)對(duì)每一頁(yè)或者某個(gè)頁(yè)面設(shè)置不同的水印效果,需要的可以參考一下2022-02-02Java精品項(xiàng)目瑞吉外賣(mài)之新增菜品與分頁(yè)查詢(xún)篇
這篇文章主要為大家詳細(xì)介紹了java精品項(xiàng)目-瑞吉外賣(mài)訂餐系統(tǒng),此項(xiàng)目過(guò)大,分為多章獨(dú)立講解,本篇內(nèi)容為新增菜品和分頁(yè)查詢(xún)功能的實(shí)現(xiàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05spring實(shí)現(xiàn)靜態(tài)注入(類(lèi)或者屬性)操作示例
這篇文章主要為大家介紹了spring實(shí)現(xiàn)靜態(tài)注入(類(lèi)或者屬性)操作示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Java中調(diào)用SQL Server存儲(chǔ)過(guò)程詳解
這篇文章主要介紹了Java中調(diào)用SQL Server存儲(chǔ)過(guò)程詳解,本文講解了使用不帶參數(shù)的存儲(chǔ)過(guò)程、使用帶有輸入?yún)?shù)的存儲(chǔ)過(guò)程、使用帶有輸出參數(shù)的存儲(chǔ)過(guò)程、使用帶有返回狀態(tài)的存儲(chǔ)過(guò)程、使用帶有更新計(jì)數(shù)的存儲(chǔ)過(guò)程等操作實(shí)例,需要的朋友可以參考下2015-01-01NameNode?重啟恢復(fù)數(shù)據(jù)的流程詳解
這篇文章主要為大家介紹了NameNode?重啟恢復(fù)數(shù)據(jù)的流程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02詳解SpringBoot定時(shí)任務(wù)說(shuō)明
本篇文章主要介紹了詳解SpringBoot定時(shí)任務(wù)說(shuō)明,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03Java使用JSON實(shí)現(xiàn)處理中文亂碼和Date格式
這篇文章主要為大家詳細(xì)介紹了Java如何在項(xiàng)目中使用JSON實(shí)現(xiàn)處理中文亂碼和Date格式的功能,文中的示例代碼講解詳細(xì),需要的小伙伴可以參考一下2023-06-06