.NET 結果與錯誤處理利器 FluentResults

来源:https://www.cnblogs.com/1312mn/p/18336221
-Advertisement-
Play Games

前言 在項目開發中,方法返回的結果(成功或失敗)對我們開發來說很重要。傳統方法,如通過異常來指示錯誤或使用特定的返回類型(如布爾值加輸出參數),雖然有效,但可能缺乏直觀性和靈活性。 FluentResults庫應運而生,它以一種既流暢又富有表達力的方式,極大地優化了這一過程。通過使用FluentRe ...


前言

在項目開發中,方法返回的結果(成功或失敗)對我們開發來說很重要。傳統方法,如通過異常來指示錯誤或使用特定的返回類型(如布爾值加輸出參數),雖然有效,但可能缺乏直觀性和靈活性。

FluentResults庫應運而生,它以一種既流暢又富有表達力的方式,極大地優化了這一過程。通過使用FluentResults,能夠以一種更加自然和易於理解的方式傳遞操作結果,包括成功狀態、錯誤信息、警告以及額外信息,提高代碼的可讀性和可維護性。

這種方式不僅讓錯誤處理更加集中和一致,還使得代碼結構更加清晰,邏輯更加流暢。

項目介紹

FluentResults 是一個在 .NET 環境中廣泛使用的庫,它提供了一種優雅的方式來處理方法執行的結果和錯誤。

使用 FluentResults,可以很容易地創建包含成功值、錯誤、警告或信息的對象,並通過鏈式調用來處理這些對象。

那麼如何使用 FluentResults 來優雅地處理結果和錯誤信息呢?

使用 FluentResults

1、安裝 FluentResults

首先,在項目中安裝 FluentResults,可以通過 NuGet 包管理器來安裝。在 Visual Studio 中也可以通過 NuGet 包管理器控制台輸入以下命令:

Install-Package FluentResults

或者,在項目文件中添加 NuGet 包引用。

2、創建 Result 對象

使用 Result 類的靜態方法來創建結果對象。Result 類提供了多種方法來創建不同類型的結果,例如成功、失敗、帶有警告或信息的成功等。

using FluentResults;  
​
static  void Main(string[] args)
{
     var result = IsInteger("");
​
     if (result.IsSuccess)
     {
         Console.WriteLine($"結果:{result.Value} ");
     }
     else
     {
         Console.WriteLine($"結果:{result.Reasons[0].Message}|{result.Errors[0].Message}");
     }
}
​
public static Result<int>  IsInteger(string input)
{
     // 假設輸入為空或null,我們可以選擇認為它不是數字  
     if (string.IsNullOrWhiteSpace(input))
     {
         return Result.Fail<int>("輸入為空或null,無法判斷是否是數字");
     }
     // 使用int.TryParse嘗試將輸入轉換為整數  
     // 如果轉換成功,out參數將包含轉換後的值,方法返回true  
     // 如果轉換失敗,方法返回false  
     if (int.TryParse(input, out int result))
     {
         return Result.Ok(result);
     }
     // 如果無法轉換為整數,則認為輸入不是數字  
     return Result.Fail<int>("輸入不是數字");
}

運行結果

通過使用Result 類我們可以看到,方法運行返回了標準的介面參數,包括IsSuccess,Message,Errors等參數,幫我們快速實現返回結構。

3、鏈式處理結果

FluentResults 允許你通過鏈式調用來處理結果,這使得錯誤處理和邏輯流程更加清晰和直觀。

需要註意的是FluentResults 本身的 Result 類型並不直接提供 OnSuccessOnFailure 這樣的鏈式方法,因為這些方法可能是在 FluentResults 的某個版本中以擴展方法的形式添加的,或者是在基於 FluentResults 的自定義擴展中定義的。

自定義擴展類

  /// <summary>
   /// Result 擴展方法
   /// </summary>
   public static class ResultExtensions
   {
       /// <summary>
       /// 成功回調
       /// </summary>
       /// <param name="result"></param>
       /// <param name="successAction"></param>
       /// <returns></returns>
       public static Result OnSuccess(this Result result, Action successAction)
       {
           if (result.IsSuccess)
           {
               successAction?.Invoke();
           }
           return result; // 返回結果以支持鏈式調用  
       }
​
       /// <summary>
       /// 失敗回調
       /// </summary>
       /// <param name="result"></param>
       /// <param name="failureAction"></param>
       /// <returns></returns>
       public static Result OnFailure(this Result result, Action<IError> failureAction)
       {
           if (!result.IsSuccess && result.Errors!= null)
           {
               foreach (var error in result.Errors)
               {
                   failureAction?.Invoke(error);
               }
           }
           return result; // 返回結果以支持鏈式調用  
       }
   }

自定義方法

/// <summary>
/// 驗證輸入字元串是否為整數
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public static Result IsIntegerInfo(string input)
{
    // 假設輸入為空或null,我們可以選擇認為它不是數字  
    if (string.IsNullOrWhiteSpace(input))
    {
        return Result.Fail("輸入為空或null,無法判斷是否是數字");
    }
    // 使用int.TryParse嘗試將輸入轉換為整數  
    // 如果轉換成功,out參數將包含轉換後的值,方法返回true  
    // 如果轉換失敗,方法返回false  
    if (int.TryParse(input, out int result))
    {
        return Result.Ok();
    }
    // 如果無法轉換為整數,則認為輸入不是數字  
    return Result.Fail("輸入不是數字");
}

調用示例

 var result = IsIntegerInfo("")
     .OnSuccess(() =>
     {
             // 處理成功的情況  
             Console.WriteLine("Success!");
     })
     .OnFailure(error =>
     {
         // 處理失敗的情況  
         Console.WriteLine("Failed: " + error.Message);
     });
 // 註意:在 OnSuccess 或 OnFailure 中使用 result 變數可能不是安全的,  
 // 因為這些回調可能在這些回調執行之前就被修改了。  
 // 更好的做法是在 OnSuccess/OnFailure 的 lambda 表達式中使用局部變數。

運行結果

 

在這個示例中定義了兩個擴展方法 OnSuccessOnFailure,它們分別接受成功和失敗時要執行的回調函數。這些方法首先檢查 Result 對象的狀態,然後根據狀態調用相應的回調函數。最後,它們返回原始的 Result 對象,以支持鏈式調用。

請註意,示例是為了說明目的而簡化的,並且可能不包含 FluentResults 庫中實際可用的所有功能和優化。在實際應用中,應該查看 FluentResults 的文檔和源代碼,以瞭解提供的具體功能。

4、FluentResults 高級特性

FluentResults提供許多高級特性,如鏈式調用、自定義錯誤類型、以及包含額外數據和元數據的錯誤對象。

例如,可以使用Result.Fail的重載版本來包含更多的上下文信息

return Result.Fail("輸入錯誤.").WithError("The input value must be greater than zero.");

5、自定義 Result 類型

FluentResults 還支持通過繼承 Result 類來創建自定義的結果類型,以便在結果中攜帶額外的數據或狀態。

public class CommonResult
{
    public Result Result { get; }
    public string MyData { get; }
​
    public CommonResult(Result result, string myData)
    {
        Result = result;
        MyData = myData;
        Console.WriteLine($"{nameof(CommonResult)}: {MyData}|{result}");
    }
}

調用示例

 public static CommonResult DemoResult(string input)
 {
     bool isSuccess =false;
     string errorMessage = "輸入的字元串不是數字";
     string myData = "測試一下";
​
     Result result = isSuccess ? Result.Ok() : Result.Fail(errorMessage);
     return new CommonResult(result, myData);
 }

運行結果

通過以上步驟,可以在 .NET 應用快速、方便的使用 FluentResults 來處理結果和錯誤。可以提高代碼的可讀性和可維護性,還可以使錯誤處理更加集中和統一規範。

使用場景

  • API 開發:在處理 HTTP 請求和響應時,FluentResults 構建清晰、一致和易於理解的錯誤響應。
  • 業務邏輯驗證:在執行業務邏輯驗證時,FluentResults 可以驗證多個錯誤,並一次性返回。
  • 複雜操作的結果處理:當需要處理包含多個步驟的複雜操作時,FluentResults 可以幫助管理每個步驟的結果,並將它們組合成一個最終的結果。

總結

FluentResults 提供了豐富的 API,可以靈活使用,與現有的 .NET 代碼庫和框架集成,如 ASP.NET Core、Entity Framework 等,還可以與其他第三方庫一起使用,以提供更全面的錯誤處理和結果功能。

如果你的項目中需要一種更好的方式來處理結果,並希望提高代碼的可讀性和可維護性,那麼 FluentResults 是一個不錯的選擇。

開源地址

https://github.com/altmann/FluentResults

如果覺得這篇文章對你有用,歡迎加入微信公眾號 [DotNet技術匠] 社區,與其他熱愛技術的同行交流心得,共同成長。


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

-Advertisement-
Play Games
更多相關文章
  • 前言 這算是一篇學習記錄博客了,主要是學習語義內核(Semantic Kernel)的實踐,以及Aspire進行全棧開發的上手體驗,我是採用Aspire同時啟動API服務,Blazor前端服務以及WinUI的桌面端項目,同時進行三個項目的代碼修改,整體感覺很方便,如果代碼都修改了只需要啟動Aspir ...
  • 或許你接觸過Jenkins, 在我理解就是拉取源碼,然後構建成鏡像,最後啟動容器! 但是這個功能對於小記憶體的伺服器來說就是奢望了! 今天介紹一個新版本,把你這個遺憾彌補下! 在PasteSpider中,也是支持拉取源碼,然後編譯發佈的!!! 以下案例使用svn作為源碼管理 如果你使用git作為源碼管 ...
  • 前言 在AspnetCore生態系統中,我們測試項目一般使用Microsoft.AspNetCore.TestHost的TestServer 到.NET6後提供的Microsoft.AspNetCore.Mvc.Testing的WebApplicationFactory,後者是前者的封裝,專門用於測 ...
  • 在.Net中,資源回收主要是指記憶體管理和非托管資源的釋放。分別提供了兩種主要的方式進行處理: 垃圾回收(GC) 確認性資源釋放(DRD) 官網相關文檔的鏈接:https://learn.microsoft.com/zh-cn/dotnet/standard/managed-code 垃圾回收(Gar ...
  • 目錄Blazor 的關鍵概念項目模板Razor 語法依賴註入註入配置HeadOutlet 組件@code 分離Blazor 調試CSS 隔離調用JavaScript 最近在學習 Blazor ,在B站上找了一個國外的課程邊看邊學習。嗯,原價¥1503的課程,大概200多美元,課程鏈接如下: B站(大 ...
  • 前言​ 推薦一個基於.NET 8 實現的通用許可權開發框架Admin.NET,前端使用Vue3/Element-plus開發。 基於.NET 8(Furion)/SqlSugar實現的通用管理平臺。整合最新技術,模塊插件式開發,前後端分離,開箱即用。 集成多租戶、緩存、數據校驗、鑒權、事件匯流排、動態A ...
  • 最近遇到一個項目,由於客戶指定了印表機型號:HP LaserJet Pro M501n ,本來想著參考網上的代碼,使用System.Management來實現對印表機狀態的查詢,測試過程中發現無論是缺紙、空閑,獲取到的統統都是空閑狀態,咨詢售後也是說無法獲取狀態,也不提供SDK;後面在社區看見可以通 ...
  • 經過2個月的詳細調研,**稀土目前缺少生產車間之間數據協同交互、缺少完整的生產工序數據協同監測和分析,無法及時瞭解生產過程和經營情況,更無法進行有效的生產過程優化。本項目推動**稀土生產環節的數字化、信息化和智能化改造,從行業、戰略、技術和市場等方案全面提升**稀土的競爭力,打造**稀土焙燒、水浸、... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...