溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

.NET?6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理

發(fā)布時間:2021-12-28 17:32:04 來源:億速云 閱讀:172 作者:柒染 欄目:開發(fā)技術(shù)

本篇文章為大家展示了.NET 6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理,內(nèi)容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。

需求

因為在項目中,會有各種各樣的領(lǐng)域異常或系統(tǒng)異常被拋出來,那么在Controller里就需要進行完整的try-catch捕獲,并根據(jù)是否有異常拋出重新包裝返回值。這是一項機械且繁瑣的工作。有沒有辦法讓框架自己去做這件事呢?

有的,解決方案的名稱叫做全局異常處理,或者叫做如何讓接口優(yōu)雅地失敗。

目標(biāo)

我們希望將異常處理和消息返回放到框架中進行統(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)全局異常處理了。接下來我們看如何完全自定義一個全局異常處理的中間件,其實原理是完全一樣的,只不過我更偏向自定義中間件的代碼組織方式,更加簡潔和一目了然。

與此同時,我們希望對返回值進行格式上的統(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)
        {
            // 你可以在這里進行相關(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:

請求

.NET?6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理

響應(yīng)

.NET?6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理

順便去看下正常返回的格式是否按我們預(yù)期的返回,下面是請求所有TodoList集合的接口返回:

.NET?6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理

可以看到正常和異常的返回類型已經(jīng)統(tǒng)一了。

其實實現(xiàn)全局異常處理還有一種方法是通過Filter來做,具體方法可以參考這篇文章:Filters in ASP.NET Core,我們之所以不選擇Filter而使用Middleware主要是基于簡單、易懂,并且作為中間件管道的第一個個中間件加入,有效地覆蓋包括中間件在內(nèi)的所有組件處理過程。Filter的位置是在路由中間件作用之后才被調(diào)用到。實際使用中,兩種方式都有應(yīng)用。 

上述內(nèi)容就是.NET 6開發(fā)TodoList應(yīng)用中如何實現(xiàn)全局異常處理,你們學(xué)到知識或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識儲備,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI