併發編程 --- 非同步方法的異常處理

来源:https://www.cnblogs.com/pandefu/archive/2023/07/31/17536263.html
-Advertisement-
Play Games

## 引言 現在模擬一個非同步方法拋出了異常: ```csharp public static async Task ThrowAfter(int ms, string message) { await Task.Delay(ms); throw new Exception(message); } ` ...


引言

現在模擬一個非同步方法拋出了異常:

public static async Task ThrowAfter(int ms, string message)
{
    await Task.Delay(ms);
    throw new Exception(message);
}

思考一下, DontHandle() 方法是否能夠捕獲到異常?

public static void DontHandle()
{
    try
    {
        ThrowAfter(1000, "first");
    }
    catch (Exception ex)
    {
        Console.WriteLine(ex.Message);
    }

}

答案是:不會捕獲到異常!

因為 DontHandle() 方法在 ThrowAfter() 方法拋出異常之前,就已經執行完畢。

非同步方法的異常處理

那麼上述代碼怎麼才能捕獲到異常呢?

若想要捕獲異常則必須通過 await 關鍵字等待 ThrowAfter() 方法執行完成。

將上文中的代碼段進行修改:

public static async void HandleoOnError()
{
    try
    {
        await ThrowAfter(1000, "first");
    }
    catch (Exception ex)
    {
        Console.WriteLine(ex.Message);
    }
}

結果就會輸出:

first

多個非同步方法的異常處理

如果調用兩個非同步方法,每個都會拋出異常,該如何處理呢?

我們可以這樣寫:

public static async void StartTwoTasks()
{
    try
    {
        await ThrowAfter(1000, "first");
        await ThrowAfter(1000, "second");
        Console.WriteLine("StartTwoTasks is Complate");
    }
    catch (Exception ex)
    {
        Console.WriteLine(ex.Message);
    }
}

思考一下輸出是什麼?

答案是:

first

並沒有預想中的兩個異常都捕獲列印出來,也沒有看到“StartTwoTasks is Complate”這句話列印出來。因為使用 await 關鍵字之後,兩次調用 ThrowAfter() 方法就變成了同步執行,捕獲到第一次的異常之後直接進入到 catch 代碼段,不再執行後續代碼。

可以嘗試解決這個問題,使用 Task.WhenAll() 方法,該方法不管任務是否拋出異常,都會等到兩個任務完成。如下代碼:

public static async void StartTwoTasksParallel()
{
    try
    {
        Task t1 = ThrowAfter(1000, "first");
        Console.WriteLine("t1 is Complate");
        Task t2 = ThrowAfter(1000, "second");
        Console.WriteLine("t2 is Complate");
        await Task.WhenAll(t2, t1);
    }
    catch (Exception ex)
    {
        Console.WriteLine(ex.Message);
    }
}

輸出:

t1 is Complate
t2 is Complate
second

從輸出可以看出來,使用 WhenAll() 方法,兩個任務都是執行完成的,但是,捕獲異常只能捕獲 WhenAll()方法參數中,排在最前面的,且第一個拋出異常的任務的消息,

上述方式有缺陷,只能拋出一個異常的任務的消息,可以將上面的方式再進化一下,如下代碼:

public static async void StartTwoTasksParallelEx()
{
    Task t1 = null;
    Task t2 = null;
    try
    {
        t1 = ThrowAfter(1000, "first");
        t2 = ThrowAfter(1000, "second");
        await Task.WhenAll(t2, t1);

    }
    catch (Exception ex)
    {
        if (t1.IsFaulted)
        {
            Console.WriteLine(t1.Exception.InnerException.Message);
        }

        if (t2.IsFaulted)
        {
            Console.WriteLine(t2.Exception.InnerException.Message);
        }
    }
}

輸出:

first
second

try/catch 代碼塊外聲明任務變數t1、t2,使他們可以在 try/catch 塊內訪問,在這裡,使用了IsFaulted 屬性,檢查任務的狀態,若IsFaulted 屬性為 true ,則表示該任務出現異常,就可以使用 Task.Exception.InnerException 訪問異常本身。

使用AggregateException信息

除了上述方式外,還有一種更好的獲取所有任務的異常信息的方式,Task.WhenAll() 方法返回的結果其實也是一個 Task 對象,而 Task 有一個 Exception 屬性,它的類型是 AggregateException,是 Exception的一個派生類,AggregateException 類有一個 InnerExceptions 屬性(異常集合,包含 Task.WhenAll() 方法列表中所有異常任務的異常信息)。

有了這個屬性則可以輕鬆遍歷所有異常。如下代碼:

public static async void StartTwoTasksParallelEx2()
{
    Task t3 = null;
    try
    {
        Task t1 = ThrowAfter(1000, "first");
        Task t2 = ThrowAfter(1000, "second");
        await (t3 = Task.WhenAll(t2, t1));

    }
    catch (Exception ex)
    {
        foreach (var item in t3.Exception.InnerExceptions)
        {
            Console.WriteLine("InnerException:" + item.Message);
        }
    }
}

輸出:

InnerException:second
InnerException:first

總結

除了前面提到的非同步方法異常處理的基本知識點,以下是一些進階的異常處理技巧:

  • 在非同步方法中,如果需要將異常傳遞給調用方,請不要直接拋出異常。相反,應該使用 throw 關鍵字將異常包裝在一個 TaskValueTask 對象中,並將其返回給調用方。這可以避免在非同步操作中丟失異常信息。

  • 如果需要在非同步方法中處理多個異常,可以使用 catch 塊來捕獲不同類型的異常,並根據需要執行不同的處理操作。還可以使用 finally 塊來執行清理操作,例如釋放資源或恢復狀態。

  • 如果需要在非同步方法中執行一些非同步操作,並且這些操作都必須成功才能繼續執行下一步操作,那麼可以使用 Task.WhenAll 方法來等待所有非同步操作完成。如果任何一個非同步操作失敗,WhenAll 方法將返回一個 AggregateException 對象,其中包含所有失敗的異常。

  • 如果需要在非同步方法中執行多個非同步操作,並且這些操作中的任何一個失敗都將導致整個操作失敗,那麼可以使用 Task.WhenAny 方法來等待第一個非同步操作完成。如果第一個操作失敗,WhenAny 方法將返回一個 AggregateException 對象,其中包含第一個失敗的異常。

  • 如果需要在非同步方法中進行錯誤處理並且希望能夠獲取更多有關異常的信息,可以使用 ExceptionDispatchInfo 類。這個類可以捕獲異常並將其存儲在一個對象中,然後在需要時重新拋出異常。這可以幫助在非同步操作中保留異常信息,並將其傳遞給調用方。

總之,在非同步方法中處理異常時,需要註意一些細節和技巧,例如正確處理異常、捕獲多個異常、等待多個非同步操作、以及使用 ExceptionDispatchInfo 類來捕獲異常。掌握這些處理技巧可以幫助編寫更可靠、更健壯的非同步代碼。

作者: Niuery Daily

出處: https://www.cnblogs.com/pandefu/>

郵箱: [email protected]

關於作者:.Net Framework,.Net Core ,WindowsForm,WPF ,控制項庫,多線程

本文版權歸作者所有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出 原文鏈接,否則保留追究法律責任的權利。 如有問題, 可郵件咨詢。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 報錯Error creating bean with name 'entityManagerFactory' defined in class path resource [org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConf... ...
  • 經過一段時間的準備,新的一期【ASP.NET Core MVC開發實戰之商城系統】已經開始,在之前的文章中,講解了商城系統的整體功能設計,頁面佈局設計,環境搭建,系統配置,及首頁【商品類型,banner條,友情鏈接,降價促銷,新品爆款】,商品列表頁面等功能的開發,今天繼續講解商品詳情功能開發,僅供學... ...
  • # Unity IPostprocessBuild技術文章 Unity IPostprocessBuild是Unity引擎中的一個非常有用的功能,它可以讓開發者在構建項目後自動執行一些操作。這個功能可以幫助開發者提高工作效率,減少手動操作的時間和錯誤率。在本文中,我們將介紹Unity IPostpr ...
  • ## 一:背景 ### 1. 講故事 前段時間有位朋友找到我,說他程式CPU直接被打滿了,讓我幫忙看下怎麼回事,截圖如下: ![](https://img2023.cnblogs.com/blog/214741/202307/214741-20230731153115090-546047217.pn ...
  • # .Net 設置 SplashScreen,在高 DPI 下不居中的解決方案 根據.Net 官方文檔,設置圖片類型的軟體的啟動屏幕非常簡單,只需要將圖片放在工程里,並將圖片的生成操作設置為 SplashScreen 後即可。 但是這個辦法在高 DPI 屏幕中沒有適應,圖片顯示的位置不居中,對於強迫 ...
  • 【前言】 本文講述的“資料庫負載均衡”方案,為市面上最經典(沒有之一),由.NET界骨灰級大佬推出。採用該技術方案的大公司,一年省下了幾個億的支出。 【正文】 支持.Net Core(2.0及以上) 與 .Net Framework(4.5及以上) 可以部署在Docker, Windows, Lin ...
  • C#泛型編程是.NET framework 2.0的新特征,它提高了代碼安全性和代碼重用的問題。由於泛型必須指定類型,正確的使用可以避免了裝箱拆箱的問題;使用了泛型後,可以將演算法或組件泛型化,基於類型參數支持任意數據類型,從而提高了代碼的重用性。 C#泛型編程在C#是很重要的特性,建議.NET開發熟 ...
  • 這篇文章介紹了什麼是事件,以及如何在C#中使用事件。事件是在生活中發生的事情,它涉及到事件的發佈者和事件的訂閱者,當事件發生時,發佈者會發佈消息,訂閱者會接收通知並做出相應的處理。在C#中,我們可以使用event關鍵字定義一個事件,然後訂閱和取消事件的方法與委托鏈的取消和鏈接相同。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...