網(wǎng)站首頁 編程語言 正文
需求
因為在項目中,會有各種各樣的領(lǐng)域異?;蛳到y(tǒng)異常被拋出來,那么在Controller
里就需要進(jìn)行完整的try-catch
捕獲,并根據(jù)是否有異常拋出重新包裝返回值。這是一項機械且繁瑣的工作。有沒有辦法讓框架自己去做這件事呢?
有的,解決方案的名稱叫做全局異常處理,或者叫做如何讓接口優(yōu)雅地失敗。
目標(biāo)
我們希望將異常處理和消息返回放到框架中進(jìn)行統(tǒng)一處理,擺脫Controller
層的try-catch
塊。
原理和思路
一般而言用來實現(xiàn)全局異常處理的思路有兩種,但是出發(fā)點都是通過.NET Web API
的管道中間件Middleware Pipeline
實現(xiàn)的。第一種方式是通過.NET
內(nèi)建的中間件來實現(xiàn);第二種是完全自定義中間件實現(xiàn)。
我們會簡單地介紹一下如何通過內(nèi)建中間件實現(xiàn),然后實際使用第二種方式來實現(xiàn)我們的代碼,大家可以比較一下異同。
在Api
項目中創(chuàng)建Models
文件夾并創(chuàng)建ErrorResponse
類。
ErrorResponse.cs
using System.Net; using System.Text.Json; namespace TodoList.Api.Models; public class ErrorResponse { public HttpStatusCode StatusCode { get; set; } = HttpStatusCode.InternalServerError; public string Message { get; set; } = "An unexpected error occurred."; public string ToJsonString() => JsonSerializer.Serialize(this); }
創(chuàng)建Extensions文件夾并新建一個靜態(tài)類ExceptionMiddlewareExtensions實現(xiàn)一個靜態(tài)擴展方法:
ExceptionMiddlewareExtensions.cs
using System.Net; using Microsoft.AspNetCore.Diagnostics; using TodoList.Api.Models; namespace TodoList.Api.Extensions; public static class ExceptionMiddlewareExtensions { public static void UseGlobalExceptionHandler(this WebApplication app) { app.UseExceptionHandler(appError => { appError.Run(async context => { context.Response.ContentType = "application/json"; var errorFeature = context.Features.Get<IExceptionHandlerFeature>(); if (errorFeature != null) { await context.Response.WriteAsync(new ErrorResponse { StatusCode = (HttpStatusCode)context.Response.StatusCode, Message = errorFeature.Error.Message }.ToJsonString()); } }); }); } }
在中間件配置的最開始配置好,注意中間件管道是有順序的,把全局異常處理放到第一步(同時也是請求返回的最后一步)能確保它能攔截到所有可能發(fā)生的異常。即這個位置:
var app = builder.Build(); app.UseGlobalExceptionHandler();
就可以實現(xiàn)全局異常處理了。接下來我們看如何完全自定義一個全局異常處理的中間件,其實原理是完全一樣的,只不過我更偏向自定義中間件的代碼組織方式,更加簡潔和一目了然。
與此同時,我們希望對返回值進(jìn)行格式上的統(tǒng)一包裝,于是定義了這樣的返回類型:
ApiResponse.cs
using System.Text.Json; namespace TodoList.Api.Models; public class ApiResponse<T> { public T Data { get; set; } public bool Succeeded { get; set; } public string Message { get; set; } public static ApiResponse<T> Fail(string errorMessage) => new() { Succeeded = false, Message = errorMessage }; public static ApiResponse<T> Success(T data) => new() { Succeeded = true, Data = data }; public string ToJsonString() => JsonSerializer.Serialize(this); }
實現(xiàn)
在Api項目中新建Middlewares文件夾并新建中間件GlobalExceptionMiddleware
GlobalExceptionMiddleware.cs
using System.Net; using TodoList.Api.Models; namespace TodoList.Api.Middlewares; public class GlobalExceptionMiddleware { private readonly RequestDelegate _next; public GlobalExceptionMiddleware(RequestDelegate next) { _next = next; } public async Task InvokeAsync(HttpContext context) { try { await _next(context); } catch (Exception exception) { // 你可以在這里進(jìn)行相關(guān)的日志記錄 await HandleExceptionAsync(context, exception); } } private async Task HandleExceptionAsync(HttpContext context, Exception exception) { context.Response.ContentType = "application/json"; context.Response.StatusCode = exception switch { ApplicationException => (int)HttpStatusCode.BadRequest, KeyNotFoundException => (int)HttpStatusCode.NotFound, _ => (int)HttpStatusCode.InternalServerError }; var responseModel = ApiResponse<string>.Fail(exception.Message); await context.Response.WriteAsync(responseModel.ToJsonString()); } }
這樣我們的ExceptionMiddlewareExtensions就可以寫成下面這樣了:
ExceptionMiddlewareExtensions.cs
using TodoList.Api.Middlewares; namespace TodoList.Api.Extensions; public static class ExceptionMiddlewareExtensions { public static WebApplication UseGlobalExceptionHandler(this WebApplication app) { app.UseMiddleware<GlobalExceptionMiddleware>(); return app; } }
驗證
首先我們需要在Controller中包裝我們的返回值,舉一個CreateTodoList的例子,其他的類似修改:
TodoListController.cs
[HttpPost] public async Task<ApiResponse<Domain.Entities.TodoList>> Create([FromBody] CreateTodoListCommand command) { return ApiResponse<Domain.Entities.TodoList>.Success(await _mediator.Send(command)); }
還記得我們在TodoList的領(lǐng)域?qū)嶓w上有一個Colour的屬性嗎,它是一個值對象,并且在賦值的過程中我們讓它有機會拋出一個UnsupportedColourException,我們就用這個領(lǐng)域異常來驗證全局異常處理。
為了驗證需要,我們可以對CreateTodoListCommand做一些修改,讓它接受一個Colour的字符串,相應(yīng)修改如下:
CreateTodoListCommand.cs
public class CreateTodoListCommand : IRequest<Domain.Entities.TodoList> { public string? Title { get; set; } public string? Colour { get; set; } } // 以下代碼位于對應(yīng)的Handler中,省略其他... var entity = new Domain.Entities.TodoList { Title = request.Title, Colour = Colour.From(request.Colour ?? string.Empty) };
啟動Api項目,我們試圖以一個不支持的顏色來創(chuàng)建TodoList:
請求
響應(yīng)
順便去看下正常返回的格式是否按我們預(yù)期的返回,下面是請求所有TodoList集合的接口返回:
可以看到正常和異常的返回類型已經(jīng)統(tǒng)一了。
總結(jié)
其實實現(xiàn)全局異常處理還有一種方法是通過Filter來做,具體方法可以參考這篇文章:Filters in ASP.NET Core,我們之所以不選擇Filter而使用Middleware主要是基于簡單、易懂,并且作為中間件管道的第一個個中間件加入,有效地覆蓋包括中間件在內(nèi)的所有組件處理過程。Filter的位置是在路由中間件作用之后才被調(diào)用到。實際使用中,兩種方式都有應(yīng)用。
下一篇我們來實現(xiàn)PUT請求。
參考資料
原文鏈接:https://www.cnblogs.com/code4nothing/archive/2021/12/27/build-todolist-8.html
相關(guān)推薦
- 2022-07-22 Android Studio Arctic Fox 的不同之處
- 2022-11-20 WPF實現(xiàn)自帶觸控鍵盤的文本框_C#教程
- 2022-07-15 QT中QByteArray與char、int、float之間的互相轉(zhuǎn)化_C 語言
- 2023-05-13 Python?readline()和readlines()函數(shù)實現(xiàn)按行讀取文件_python
- 2022-10-14 如何查看EJB項目的版本2.0或3.0
- 2022-03-28 C++讀取wav文件中的PCM數(shù)據(jù)_C 語言
- 2022-01-10 npm一個錯誤 npm ERR code ENOENT npm ERR syscall open
- 2022-09-18 IOS開發(fā)壓縮后圖片模糊問題解決_IOS
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支