千家信息网

.NET 6开发TodoList应用中如何实现全局异常处理

发表于:2025-02-23 作者:千家信息网编辑
千家信息网最后更新 2025年02月23日,本篇文章为大家展示了.NET 6开发TodoList应用中如何实现全局异常处理,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。需求因为在项目中,会有各种各样的领
千家信息网最后更新 2025年02月23日.NET 6开发TodoList应用中如何实现全局异常处理

本篇文章为大家展示了.NET 6开发TodoList应用中如何实现全局异常处理,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。

需求

因为在项目中,会有各种各样的领域异常或系统异常被抛出来,那么在Controller里就需要进行完整的try-catch捕获,并根据是否有异常抛出重新包装返回值。这是一项机械且繁琐的工作。有没有办法让框架自己去做这件事呢?

有的,解决方案的名称叫做全局异常处理,或者叫做如何让接口优雅地失败。

目标

我们希望将异常处理和消息返回放到框架中进行统一处理,摆脱Controller层的try-catch块。

原理和思路

一般而言用来实现全局异常处理的思路有两种,但是出发点都是通过.NET Web API的管道中间件Middleware Pipeline实现的。第一种方式是通过.NET内建的中间件来实现;第二种是完全自定义中间件实现。

我们会简单地介绍一下如何通过内建中间件实现,然后实际使用第二种方式来实现我们的代码,大家可以比较一下异同。

Api项目中创建Models文件夹并创建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);}

创建Extensions文件夹并新建一个静态类ExceptionMiddlewareExtensions实现一个静态扩展方法:

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();                if (errorFeature != null)                {                    await context.Response.WriteAsync(new ErrorResponse                    {                        StatusCode = (HttpStatusCode)context.Response.StatusCode,                        Message = errorFeature.Error.Message                    }.ToJsonString());                }            });        });    }}

在中间件配置的最开始配置好,注意中间件管道是有顺序的,把全局异常处理放到第一步(同时也是请求返回的最后一步)能确保它能拦截到所有可能发生的异常。即这个位置:

var app = builder.Build();app.UseGlobalExceptionHandler();

就可以实现全局异常处理了。接下来我们看如何完全自定义一个全局异常处理的中间件,其实原理是完全一样的,只不过我更偏向自定义中间件的代码组织方式,更加简洁和一目了然。

与此同时,我们希望对返回值进行格式上的统一包装,于是定义了这样的返回类型:

ApiResponse.cs

using System.Text.Json;namespace TodoList.Api.Models;public class ApiResponse{    public T Data { get; set; }    public bool Succeeded { get; set; }    public string Message { get; set; }    public static ApiResponse Fail(string errorMessage) => new() { Succeeded = false, Message = errorMessage };    public static ApiResponse Success(T data) => new() { Succeeded = true, Data = data };    public string ToJsonString() => JsonSerializer.Serialize(this);}

实现

在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)        {            // 你可以在这里进行相关的日志记录            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.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();        return app;    }}

验证

首先我们需要在Controller中包装我们的返回值,举一个CreateTodoList的例子,其他的类似修改:

TodoListController.cs

[HttpPost]public async Task> Create([FromBody] CreateTodoListCommand command){    return ApiResponse.Success(await _mediator.Send(command));}

还记得我们在TodoList的领域实体上有一个Colour的属性吗,它是一个值对象,并且在赋值的过程中我们让它有机会抛出一个UnsupportedColourException,我们就用这个领域异常来验证全局异常处理。

为了验证需要,我们可以对CreateTodoListCommand做一些修改,让它接受一个Colour的字符串,相应修改如下:

CreateTodoListCommand.cs

public class CreateTodoListCommand : IRequest{    public string? Title { get; set; }    public string? Colour { get; set; }}// 以下代码位于对应的Handler中,省略其他...var entity = new Domain.Entities.TodoList{    Title = request.Title,    Colour = Colour.From(request.Colour ?? string.Empty)};

启动Api项目,我们试图以一个不支持的颜色来创建TodoList:

请求

响应

顺便去看下正常返回的格式是否按我们预期的返回,下面是请求所有TodoList集合的接口返回:

可以看到正常和异常的返回类型已经统一了。

其实实现全局异常处理还有一种方法是通过Filter来做,具体方法可以参考这篇文章:Filters in ASP.NET Core,我们之所以不选择Filter而使用Middleware主要是基于简单、易懂,并且作为中间件管道的第一个个中间件加入,有效地覆盖包括中间件在内的所有组件处理过程。Filter的位置是在路由中间件作用之后才被调用到。实际使用中,两种方式都有应用。

上述内容就是.NET 6开发TodoList应用中如何实现全局异常处理,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注行业资讯频道。

中间件 处理 全局 方式 项目 应用 代码 文件 文件夹 方法 管道 领域 统一 验证 开发 位置 内容 原理 实际 思路 数据库的安全要保护哪些东西 数据库安全各自的含义是什么 生产安全数据库录入 数据库的安全性及管理 数据库安全策略包含哪些 海淀数据库安全审计系统 建立农村房屋安全信息数据库 易用的数据库客户端支持安全管理 连接数据库失败ssl安全错误 数据库的锁怎样保障安全 软件开发可以控制概率 大学生如何守住网络安全的红线 网络安全都用什么语言 力控如何将数据存入关系数据库 计算机网络技术的就业方向与前景 iis停止服务器 英灵神殿有没有服务器 云服务器弄成云虚拟主机 你无法在安全服务器上游戏 valorant服务器国服 公文管理系统数据库 网络安全热词 端点安全 技术人员数据库管理 中天数据库 wow服务器列表 家庭网络如何搭建网站服务器 虚荣服务器在哪里 数据库中差的运算条件 长沙软件开发工资很低吗 大专学习计算机网络技术好吗 广州买东买西互联网科技有限公司 将数据库中数据通过分批 服务器单根网线怎么做出来好看 网络安全和信息化.技术 英灵神殿有没有服务器 移动公司网络技术员做什么 网络安全热词 端点安全 贵州6网络安全教育 扫黄护苗网络安全宣传资料 导入操作怎样获得数据库的值
0