聊聊.NET Core处理全局异常有那些方法

embedded/2024/10/18 18:21:04/

简述

处理全局异常的方法有IExceptionFilter(异常处理),使用中间件异常处理,使用框架自带异常中间件等。

考点

考察对异常处理方式的熟悉程度和广度,以及对中间件、过滤器熟练程度。

下面分别具体介绍三种处理异常的方法
1、异常处理 IExceptionFilter
开发过ASP.NET程序的人都知道:IExceptionFilter。这个过滤器同样在AspNetCore中也可以用来捕获异常。不过,对于使用IExceptionFilter,更建议使用它的异步版本:IAsyncExceptionFilter。那么该如何使用过滤器呢?下面以IAsyncExceptionFilter为例,对于同步版本其实也是一样的。

我们在项目中添加一个Model文件夹,存放返回结果实体类,这里定义一个泛型类:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;namespace ExceptionDemo.Model
{public class ResultModel<T>{/// <summary>/// 返回结果编码 0:失败 1:成功/// </summary>public int ResultCode { get; set; }/// <summary>/// 返回结果内容 成功:Success  失败:异常内容/// </summary>public string ResultMsg { get; set; }/// <summary>/// 返回结果 成功:返回T类型数据 失败:默认null/// </summary>public T ResultData { get; set; }}
}

我们在项目中添加一个Filter文件夹,所有的过滤器都放在该文件夹下面。然后添加一个类:CustomerExceptionFilter,并使该类继承自IAsyncExceptionFilter。代码如下:

using ExceptionDemo.Model;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.Filters;
using Newtonsoft.Json;
using System.Threading.Tasks;namespace ExceptionDemo.Filter
{/// <summary>/// 自定义异常过滤器/// </summary>public class CustomerExceptionFilter : IAsyncExceptionFilter{/// <summary>/// 重写OnExceptionAsync方法,定义自己的处理逻辑/// </summary>/// <param name="context"></param>/// <returns></returns>private readonly ILogger<CustomerExceptionFilter> _logger;public CustomerExceptionFilter(ILogger<CustomerExceptionFilter> logger){_logger = logger;}public Task OnExceptionAsync(ExceptionContext context){// 如果异常没有被处理则进行处理if(context.ExceptionHandled==false){// 定义返回类型var result = new ResultModel<string>{ResultCode = 0,ResultMsg = context.Exception.Message};_logger.LogInformation("502 Bad Gateway: ");context.Result = new ContentResult{// 返回状态码设置为200,表示成功StatusCode = StatusCodes.Status200OK,// 设置返回格式ContentType="application/json;charset=utf-8",Content=JsonConvert.SerializeObject(result)};}// 设置为true,表示异常已经被处理了context.ExceptionHandled = true;return Task.CompletedTask;}}
}

上面的代码很简单,我们新建了一个自定义的异常过滤器,然后在OnExceptionAsync方法中定义自己的处理逻辑,报错之后依然让http返回状态码为200,并且将错误信息返回到客户端。

然后添加一个控制器,命名为ExceptionFilter,在控制器中模拟发生异常的情况:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using ExceptionDemo.Model;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;namespace ExceptionDemo.Controllers
{[Route("api/[controller]")][ApiController]public class ExceptionFilterController : ControllerBase{[HttpGet]public async Task<ResultModel<int>> Get(){int i = 0;int k = 10;// 这里会发生异常int j = await Task.Run<int>(() => {return k / i;});return new ResultModel<int>(){ResultCode=1,ResultMsg="Success",ResultData=j};}}
}

最后我们需要把自定义的异常过滤器进行注入,这里选择使用全局注入的方式,在Startup类的ConfigureServices方法中进行注入:

builder.Services.AddLogging(logging =>
{logging.AddConsole(); // 向控制台输出日志// 你可以根据需要添加其他日志提供程序,比如文件日志、事件日志等
});
services.AddControllers(options => 
{   //因为CustomerExceptionFilter构造函数中使用了ILogger 所以必须现在前面注入AddLoggingoptions.Filters.Add<CustomerExceptionFilter>();
});

然后运行程序,查看结果:
在这里插入图片描述
如何我们没有使用过滤器捕获和处理异常,我们将得到Http状态码为500的内部错误,这种错误不方便定位问题,而且给客户端返回的信息也不够友好。使用了过滤器处理异常,进行特殊处理之后就会显得很友好了。

在上面自定义过滤器的代码中,有下面的一行代码:

context.ExceptionHandled = true;
注意:这句代码很关键,当你处理完异常之后,一定要将此属性更改为true,表示异常已经处理过了,这样其他地方就不会在处理这个异常了。
2、使用中间件处理异常
我们知道,AspNetCore的管道模型具有层层传递的特点,那么我们就可以在管道中实现全局异常捕获。我们新创建一个自定义的异常中间件:

using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using System;
using System.Text.Json;
using System.Threading.Tasks;namespace ExceptionDemo.Middleware
{/// <summary>/// 自定义异常中间件/// </summary>public class CustomerExceptionMiddleware{/// <summary>/// 委托/// </summary>private readonly RequestDelegate _next;public CustomerExceptionMiddleware(RequestDelegate next){_next = next;}public async Task Invoke(HttpContext context){try{await _next(context);}catch (Exception ex){context.Response.ContentType = "application/problem+json";var title = "An error occured: " + ex.Message;var details = ex.ToString();var problem = new ProblemDetails{Status = 200,Title = title,Detail = details};var stream = context.Response.Body;await JsonSerializer.SerializeAsync(stream, problem);}}}
}

然后在新建一个扩展方法:

using Microsoft.AspNetCore.Builder;namespace ExceptionDemo.Middleware
{/// <summary>/// 静态类/// </summary>public static class ExceptionMiddlewareExtension{/// <summary>/// 静态方法/// </summary>/// <param name="app">要进行扩展的类型</param>public static void UseExceptionMiddleware(this IApplicationBuilder app){app.UseMiddleware(typeof(CustomerExceptionMiddleware));}}
}

最后在Startup类的Configure方法中使用自定义的异常中间件:

app.UseExceptionMiddleware();
然后我们注释掉上面注册的异常过滤器,运行程序进行访问:
在这里插入图片描述
这样也可以捕获到异常。
3、使用框架自带异常中间件

if (env.IsDevelopment())
{app.UseDeveloperExceptionPage();
}

这段代码在我们使用AspNetCore创建一个WebApi项目时就会看到,如果是创建的MVC项目,是下面一段代码:

if (env.IsDevelopment())
{app.UseDeveloperExceptionPage();
}
else
{app.UseExceptionHandler("/Error");
}

这两段代码的作用就是捕获和处理异常,是第一个被添加到管道中的中间件。

UseDeveloperExceptionPage的意思很好理解:对于开发模式,一旦报错就跳转到错误堆栈页面。而第二个UseExceptionHandler也很有意思,从它的名字中我们大致可以猜出它肯定是个错误拦截程序。那么它和上面自定义的异常处理中间件有什么区别呢?

UseExceptionHandler其实就是默认的错误处理。它其实也是一个中间件,它的原名叫做ExceptionHandlerMiddleware。在使用UseExceptionHandler方法时,我们可以选填各种参数。比如上面的第二段代码,填入了“/Error”参数,表示当产生异常的时候,将定位到对应的路径,这里定位的页面就是“http://localhost:5001/Error”。这是MVC中自带的一个错误页面,当然,你也可以指定自己定义的一个页面。

UseExceptionHandler还有一个指定ExceptionHandlerOptions参数的扩展方法,该参数是ExceptionHandlerMiddleware中间件的重要参数:

参数名 说明
ExceptionHandlingPath 重定向的路径,比如刚才的 “”/Error"" 实际上就是指定的该参数
ExceptionHandler 错误拦截处理程序
ExceptionHandler允许我们在ExceptionHandlerMiddleware内部指定咱们自己的异常处理逻辑。而该参数的类型为RequestDelegate类型的委托。因此,UseExceptionHandler提供了一个简便的写法,可以让我们在ExceptionHandlerMiddleware中新建自定义的错误拦截管道来处理异常:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.Json;
using System.Threading.Tasks;
using ExceptionDemo.Filter;
using ExceptionDemo.Middleware;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Diagnostics;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;namespace ExceptionDemo
{public class Startup{public Startup(IConfiguration configuration){Configuration = configuration;}public IConfiguration Configuration { get; }// This method gets called by the runtime. Use this method to add services to the container.public void ConfigureServices(IServiceCollection services){#region 注册全局异常过滤器//services.AddControllers(options => //{//    options.Filters.Add(new CustomerExceptionFilter());//});#endregionservices.AddControllers();}// This method gets called by the runtime. Use this method to configure the HTTP request pipeline.public void Configure(IApplicationBuilder app, IWebHostEnvironment env){if (env.IsDevelopment()){app.UseDeveloperExceptionPage();}else{app.UseExceptionHandler(builder => builder.Use(ExceptionHandlerDemo));}app.UseExceptionMiddleware();app.UseRouting();app.UseAuthorization();app.UseEndpoints(endpoints =>{endpoints.MapControllers();});}private async Task ExceptionHandlerDemo(HttpContext httpContext,Func<Task> next){//该信息由ExceptionHandlerMiddleware中间件提供,里面包含了ExceptionHandlerMiddleware中间件捕获到的异常信息。var exceptionDetails = httpContext.Features.Get<IExceptionHandlerFeature>();var ex = exceptionDetails?.Error;if (ex != null){httpContext.Response.ContentType = "application/problem+json";var title = "An error occured: " + ex.Message;var details = ex.ToString();var problem = new ProblemDetails{Status = 500,Title = title,Detail = details};var stream = httpContext.Response.Body;await JsonSerializer.SerializeAsync(stream, problem);}}}
}

中间件和过滤器的比较
在上面的例子中,我们分别使用了中间件和过滤器的方式来处理异常,那么中间件和过滤器有什么区别呢?两者的区别:拦截范围的不同。
IExceptionFilter作为一种过滤器,它需要在控制器发现错误之后将错误信息提交给它处理,因此它的异常处理范围是控制器内部。如果我们想捕获进入控制器之前的一些错误,IExceptionFilter是捕获不到的。而对于ExceptionHandlerMiddleware异常中间件来说就很容易了,它作为第一个中间件被添加到管道中,在它之后发生的任何异常都可以捕获的到。

那么为什么要有两种异常处理的方式呢?只使用ExceptionHandlerMiddleware中间件处理异常不可以吗?它可以捕获任何时候发生的异常,为什么还要有过滤器呢?如果你想在控制器发生异常时快速捕获和处理异常,那么使用过滤器处理异常是非常不错的选择。如果是控制器内部发生了异常,首先是由过滤器捕获到异常,最后才是中间件捕获到异常。

我们在自定义过滤器的时候有这样一段代码:context.ExceptionHandled = true;如果在自定义过滤器中将异常标记为已经处理之后,则第一个异常处理中间件就认为没有错误了,不会进入到处理逻辑中了。所以,如果不把 ExceptionHandled属性设置为true,可能出现异常处理结果被覆盖的情况。


http://www.ppmy.cn/embedded/21218.html

相关文章

买卖股票+跳跃游戏 贪心算法

买卖股票的最佳时机 只收集正利润即可&#xff01; class Solution {public int maxProfit(int[] prices) {int sum0;for(int i0;i<prices.length-1;i){if(prices[i1]-prices[i]>0){//只收集正利润sumprices[i1]-prices[i];}}return sum;} } 跳跃游戏 看覆盖范围&…

Axure中的样式

样式 首先说一下Axure里面的原点位置 如下图&#xff1a; 还有一个办法是我们选中我们的按钮&#xff0c;如上图&#xff0c;然后打开右边的样式&#xff0c;可以看按钮的x&#xff0c;y属性&#xff0c;类似于游戏中unity软件的x&#xff0c;y属性&#xff0c;类似于html中…

colab直接下载kaggle数据集

1.获取kaggle API Token account-settings-API&#xff0c;这时候会下载一个kaggle.json&#xff0c;然后将里面的内容{username:.......}复制一份 &#xff08;个人怕之后找不到另外放了一份到Google云端硬盘中&#xff09; 2.在colab中绑定kaggle !pip install -U -q kagg…

《深入浅出.NET框架设计与实现》笔记1——.NET CLI 概述

.NET CLI&#xff08;NET 命令行接口&#xff09;工具是用于开发生成运行和发布.NET应用程序的跨平台工具链。 一、CLI命令 默认安装的命令有 1、基本命令 new restore build publish run test vstest pack migrate clean sln help store 2、项目修改命令 add package add …

[Spring Cloud] (4)搭建Vue2与网关、微服务通信并配置跨域

文章目录 前言gatway网关跨域配置取消微服务跨域配置 创建vue2项目准备一个原始vue2项目安装vue-router创建路由vue.config.js配置修改App.vue修改 添加接口访问安装axios创建request.js创建index.js创建InfoApi.js main.jssecurityUtils.js 前端登录界面登录消息提示框 最终效…

【golang学习之旅】报错:a declared but not used

目录 报错原因解决方法参考 报错 代码很简单&#xff0c;如下所示。可以发现a和b都飙红了&#xff1a; 运行后就会出现报错&#xff1a; 报错翻译过来就是a已经声明但未使用。当时我很疑惑&#xff0c;在其他语言中从来没有这种情况。况且这里的b不是赋值了吗&#xff0c;怎…

mysql数据库开发军规

MySQL数据库开发军规是一系列最佳实践和原则&#xff0c;旨在帮助开发者在MySQL数据库设计和开发过程中提升性能、确保数据安全、减少错误&#xff0c;并提高可维护性。以下是一些关键的MySQL开发军规&#xff1a; 核心军规&#xff1a; 避免在数据库中进行复杂运算&#xff…

IPD项目管理体系的建立以及项目管理软件如何助力IPD高效落地

市场竞争的加剧与客户需求的多样性&#xff0c;传统的研发管理模式已无法满足企业发展的进程&#xff0c;IPD作为一种先进的研发管理思想与方法已被更多的企业所采用&#xff0c;用以提高研发效率。建立一个高效的IPD流程管理体系对于企业的发展至关重要。 接下来&#xff0c;…