iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >.NET 6开发TodoList应用中如何实现全局异常处理
  • 376
分享到

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

2023-06-22 05:06:23 376人浏览 安东尼
摘要

本篇文章为大家展示了.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<IExceptionHandlerFeature>();                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<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);}

实现

在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<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的领域实体上有一个Colour的属性吗,它是一个值对象,并且在赋值的过程中我们让它有机会抛出一个UnsupportedColourException,我们就用这个领域异常来验证全局异常处理。

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

CreateTodoListCommand.cs

public class CreateTodoListCommand : IRequest<Domain.Entities.TodoList>{    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:

请求

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

响应

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

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

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

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

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

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

--结束END--

本文标题: .NET 6开发TodoList应用中如何实现全局异常处理

本文链接: https://www.lsjlt.com/news/302937.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • c#程序自启动怎么设置
    c# 程序的自启动方法有三种:注册表:在指定注册表项下创建新值,并将其设置为程序可执行文件路径。任务计划程序:创建一个新任务,并在触发器和动作部分分别指定登录时或特定时间触发,以及启动程...
    99+
    2024-05-14
    c#
  • c#怎么调用dll文件
    可在 c# 中轻松调用 dll 文件:引用 dll(使用 dllimport 特性)定义与 dll 函数签名匹配的函数原型调用 dll 函数(如同 c# 函数)附加技巧:使用 chars...
    99+
    2024-05-14
    c#
  • 如何构建 Golang RESTful API,并实现 CRUD 操作?
    通过创建 golang 项目并安装必要的包,我们可以构建一个功能齐全的 restful api。它使用 mysql 数据库进行 crud 操作:1. 创建和连接数据库;2. 定义数据结构...
    99+
    2024-05-14
    go crud mysql git golang
  • c#怎么添加类文件
    在c#中添加类文件的步骤:1. 创建新项目,2. 添加新类,3. 为类添加代码,4. 在另一个类中引用新类。using语句引用类文件所在的命名空间;new运算符创建类的新实例;点运算符访...
    99+
    2024-05-14
    c#
  • 使用 C++ 构建高性能服务器架构的最佳实践
    遵循 c++++ 中构建高性能服务器架构的最佳实践可以创建可扩展、可靠且可维护的系统:使用线程池以重用线程,提高性能。利用协程减少上下文切换和内存开销,提升性能。通过智能指针和引用计数优...
    99+
    2024-05-14
    c++ 高性能服务器架构 数据访问
  • c#怎么添加字段
    在 c# 中添加字段包括以下步骤:声明字段:在类或结构中使用 字段类型 字段名; 语法声明字段。访问修饰符:用于限制对字段的访问,如 private、public、protected 和...
    99+
    2024-05-14
    c#
  • c#中怎么添加引用
    c# 中添加引用的方法有四种:使用 nuget 包管理器添加软件包。添加项目引用以包含其他项目。手动编辑项目文件 (.csproj) 以添加引用。从编译器命令行使用 /reference...
    99+
    2024-05-14
    c#
  • c#怎么创建文本文件
    在 c# 中创建文本文件的方法包括:创建 filestream 对象以打开或创建文件。使用 streamwriter 写入文本至文件。关闭 streamwriter 对象释放资源。关闭 ...
    99+
    2024-05-14
    c#
  • c#怎么定义属性
    如何在 c# 中定义属性 属性是一种编程构造,它包含一个 get 访问器和一个 set 访问器,允许以一种类属性的方式访问字段。它们提供了一种安全且封装的方式来访问和修改类的内部数据。 ...
    99+
    2024-05-14
    c#
  • 基于 C++ 的服务器架构的安全性考虑因素
    在设计基于 c++++ 的服务器架构时,安全考虑至关重要:使用 std::string 或 std::vector 避免缓冲区溢出。使用正则表达式或库函数验证用户输入。采用输出转义防止跨...
    99+
    2024-05-14
    安全性 关键词: c++ 服务器架构 c++ lsp
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作