.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 8、WPF、Prism.DryIoc、MVVM設計模式、Blazor以及MySQL資料庫構建的企業級工作流系統的WPF客戶端框架-AIStudio.Wpf.AClient 6.0。 項目介紹 框架採用了 Prism 框架來實現 MVVM 模式,不僅簡化了 MVVM 的典型 ...
  • 先看一下效果吧: 我們直接通過改造一下原版的TreeView來實現上面這個效果 我們先創建一個普通的TreeView 代碼很簡單: <TreeView> <TreeViewItem Header="人事部"/> <TreeViewItem Header="技術部"> <TreeViewItem He ...
  • 1. 生成式 AI 簡介 https://imp.i384100.net/LXYmq3 2. Python 語言 https://imp.i384100.net/5gmXXo 3. 統計和 R https://youtu.be/ANMuuq502rE?si=hw9GT6JVzMhRvBbF 4. 數 ...
  • 本文為大家介紹下.NET解壓/壓縮zip文件。雖然解壓縮不是啥核心技術,但壓縮性能以及進度處理還是需要關註下,針對使用較多的zip開源組件驗證,給大家提供個技術選型參考 之前在《.NET WebSocket高併發通信阻塞問題 - 唐宋元明清2188 - 博客園 (cnblogs.com)》講過,團隊 ...
  • 之前寫過兩篇關於Roslyn源生成器生成源代碼的用例,今天使用Roslyn的代碼修複器CodeFixProvider實現一個cs文件頭部註釋的功能, 代碼修複器會同時涉及到CodeFixProvider和DiagnosticAnalyzer, 實現FileHeaderAnalyzer 首先我們知道修 ...
  • 在軟體行業,經常會聽到一句話“文不如表,表不如圖”說明瞭圖形在軟體應用中的重要性。同樣在WPF開發中,為了程式美觀或者業務需要,經常會用到各種個樣的圖形。今天以一些簡單的小例子,簡述WPF開發中幾何圖形(Geometry)相關內容,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 在 C# 中使用 RabbitMQ 通過簡訊發送重置後的密碼到用戶的手機號上,你可以按照以下步驟進行 1.安裝 RabbitMQ 客戶端庫 首先,確保你已經安裝了 RabbitMQ 客戶端庫。你可以通過 NuGet 包管理器來安裝: dotnet add package RabbitMQ.Clien ...
  • 1.下載 Protocol Buffers 編譯器(protoc) 前往 Protocol Buffers GitHub Releases 頁面。在 "Assets" 下找到適合您系統的壓縮文件,通常為 protoc-{version}-win32.zip 或 protoc-{version}-wi ...
  • 簡介 在現代微服務架構中,服務發現(Service Discovery)是一項關鍵功能。它允許微服務動態地找到彼此,而無需依賴硬編碼的地址。以前如果你搜 .NET Service Discovery,大概率會搜到一大堆 Eureka,Consul 等的文章。現在微軟為我們帶來了一個官方的包:Micr ...
  • ZY樹洞 前言 ZY樹洞是一個基於.NET Core開發的簡單的評論系統,主要用於大家分享自己心中的感悟、經驗、心得、想法等。 好了,不賣關子了,這個項目其實是上班無聊的時候寫的,為什麼要寫這個項目呢?因為我單純的想吐槽一下工作中的不滿而已。 項目介紹 項目很簡單,主要功能就是提供一個簡單的評論系統 ...