(给DotNet加星标,提升.Net技能)

转自:#山鸡

cnblogs.com/ShenNan/p/10197231.html

为什么异常处理选择中间件?

传统的ASP.NET可以采用异常过滤器的方式处理异常,在ASP.NET CORE中,是以多个中间件连接而成的管道形式处理请求的,不过常用的五大过滤器得以保留,同样可以采用异常过滤器处理异常,但是异常过滤器不能处理MVC中间件以外的异常,为了全局统一考虑,采用中间件处理异常更为合适

为什么选择自定义异常中间件?

先来看看ASP.NET CORE 内置的三个异常处理中间件

1、DeveloperExceptionPageMiddleware 

能给出详细的请求/返回/错误信息,因为包含敏感信息,所以仅适合开发环境

2、ExceptionHandlerMiddleware  

仅处理500错误

3、StatusCodePagesMiddleware  

能处理400-599之间的错误,但需要Response中不能包含内容(ContentLength=0 && ContentType=null,经实验不能响应mvc里未捕获异常)

由于ExceptionHandlerMiddleware和StatusCodePagesMiddleware的各自的限制条件,两者需要搭配使用。相比之下自定义中间件更加灵活,既能对各种错误状态进行统一处理,也能按照配置决定处理方式。

CustomExceptionMiddleWare

首先声明异常中间件的配置类

/// /// 异常中间件配置对象/// public class CustomExceptionMiddleWareOption{    public CustomExceptionMiddleWareOption(        CustomExceptionHandleType handleType = CustomExceptionHandleType.JsonHandle,        IList jsonHandleUrlKeys = null,string errorHandingPath = ""){        HandleType = handleType;        JsonHandleUrlKeys = jsonHandleUrlKeys;        ErrorHandingPath = errorHandingPath;    }    ///     /// 异常处理方式    ///     public CustomExceptionHandleType HandleType { get; set; }

    ///     /// Json处理方式的Url关键字    /// 仅HandleType=Both时生效    ///     public IList JsonHandleUrlKeys { get; set; }/// /// 错误跳转页面/// public PathString ErrorHandingPath { get; set; }}/// /// 错误处理方式/// public enum CustomExceptionHandleType{    JsonHandle = 0,   //Json形式处理    PageHandle = 1,   //跳转网页处理    Both = 2          //根据Url关键字自动处理}

声明异常中间件的成员

/// /// 管道请求委托/// private RequestDelegate _next;

/// /// 配置对象/// private CustomExceptionMiddleWareOption _option;

/// /// 需要处理的状态码字典/// private IDictionary<int, string> exceptionStatusCodeDic;public CustomExceptionMiddleWare(RequestDelegate next, CustomExceptionMiddleWareOption option){    _next = next;    _option = option;    exceptionStatusCodeDic = new Dictionary<int, string>    {        { 401, "未授权的请求" },        { 404, "找不到该页面" },        { 403, "访问被拒绝" },        { 500, "服务器发生意外的错误" }        //其余状态自行扩展    };}

异常中间件主要逻辑

public async Task Invoke(HttpContext context){    Exception exception = null;    try    {        await _next(context);   //调用管道执行下一个中间件    }    catch (Exception ex)    {        context.Response.Clear();            context.Response.StatusCode = 500;   //发生未捕获的异常,手动设置状态码        exception = ex;    }    finally    {        if (exceptionStatusCodeDic.ContainsKey(context.Response.StatusCode) &&             !context.Items.ContainsKey("ExceptionHandled"))  //预处理标记        {            var errorMsg = string.Empty;            if (context.Response.StatusCode == 500 && exception != null)            {                errorMsg = $"{exceptionStatusCodeDic[context.Response.StatusCode]}\r\n{(exception.InnerException != null ? exception.InnerException.Message : exception.Message)}";         }         else         {             errorMsg = exceptionStatusCodeDic[context.Response.StatusCode];         }         exception = new Exception(errorMsg);     }     if (exception != null)     {         var handleType = _option.HandleType;         if (handleType == CustomExceptionHandleType.Both)   //根据Url关键字决定异常处理方式         {             var requestPath = context.Request.Path;             handleType = _option.JsonHandleUrlKeys != null && _option.JsonHandleUrlKeys.Count(                 k => context.Request.Path.StartsWithSegments(k, StringComparison.CurrentCultureIgnoreCase)) > 0 ?                 CustomExceptionHandleType.JsonHandle :                 CustomExceptionHandleType.PageHandle;         }         if (handleType == CustomExceptionHandleType.JsonHandle)             await JsonHandle(context, exception);         elseawait PageHandle(context, exception, _option.ErrorHandingPath);     }  }}

/// /// 统一格式响应类/// /// /// private ApiResponse GetApiResponse(Exception ex){   return new ApiResponse() { IsSuccess = false, Message = ex.Message };}

/// /// 处理方式:返回Json格式/// /// /// /// private async Task JsonHandle(HttpContext context, Exception ex){    var apiResponse = GetApiResponse(ex);    var serialzeStr = JsonConvert.SerializeObject(apiResponse);    context.Response.ContentType = "application/json";    await context.Response.WriteAsync(serialzeStr, Encoding.UTF8);}

/// /// 处理方式:跳转网页/// /// /// /// /// private async Task PageHandle(HttpContext context, Exception ex, PathString path){    context.Items.Add("Exception", ex);    var originPath = context.Request.Path;    context.Request.Path = path;   //设置请求页面为错误跳转页面    try    {        await _next(context);          }    catch { }    finally    {        context.Request.Path = originPath;   //恢复原始请求页面    }}

使用扩展类进行中间件注册

public static class CustomExceptionMiddleWareExtensions{    public static IApplicationBuilder UseCustomException(this IApplicationBuilder app, CustomExceptionMiddleWareOption option){        return app.UseMiddleware(option);    }}

在Startup.cs的Configuref方法中注册异常中间件

app.UseCustomException(new CustomExceptionMiddleWareOption(    handleType: CustomExceptionHandleType.Both,      //根据url关键字决定处理方式    jsonHandleUrlKeys: new PathString[] { "/api" },    errorHandingPath: "/home/error"));

接下来我们来进行测试,首先模拟一个将会进行页面跳转的未经捕获的异常

访问/home/about的结果

访问/home/test的结果 (该地址不存在)

OK异常跳转页面的方式测试完成,接下来我们测试返回统一格式(json)的异常处理,同样先模拟一个未经捕获的异常

访问/api/token/gettesterror的结果

访问/api/token/test的结果 (该地址不存在)

访问/api/token/getvalue的结果 (该接口需要身份验证)

测试完成,页面跳转和统一格式返回都没有问题,自定义异常中间件已按预期工作。

需要注意的是,自定义中间件会响应每个HTTP请求,所以处理逻辑一定要精简,防止发生不必要的性能问题。

推荐阅读

(点击标题可跳转阅读)

ASP.NET Core依赖注入和管道方式的异常处理

surging 微服务引擎 1.0 正式发布

三分钟学会.NET微服务之Polly

看完本文有收获?请转发分享给更多人

关注「DotNet」加星标,提升.Net技能

喜欢就点一下「好看」呗~

自定义request_ASP.NET Core 学习之自定义异常处理相关推荐

  1. java怎样实现自定义过滤关键词_SpringSecurity学习之自定义过滤器的实现代码

    我们系统中的认证场景通常比较复杂,比如说用户被锁定无法登录,限制登录IP等.而SpringSecuriy最基本的是基于用户与密码的形式进行认证,由此可知它的一套验证规范根本无法满足业务需要,因此扩展势 ...

  2. 如何在ASP.NET Core中编写自定义日志记录提供程序

    目录 介绍 如何实现所需的接口 基础类和附件 FileLoggerProvider具体类及其附件 1. ConfigureLogging() 2. appsettings.json文件 介绍 源代码可 ...

  3. Java虚拟机JVM学习06 自定义类加载器 父委托机制和命名空间的再讨论

    Java虚拟机JVM学习06 自定义类加载器 父委托机制和命名空间的再讨论 创建用户自定义的类加载器 要创建用户自定义的类加载器,只需要扩展java.lang.ClassLoader类,然后覆盖它的f ...

  4. 修改meta标签 查看源码没效果怎么办_Spring 源码学习(三)-自定义标签

    又来填坑啦,上一篇讲完默认标签的解析,这篇笔记记录一下自定义标签的解析吧. 我们知道,Spring 源码的核心模块是 Spring-core 和 Spring-beans,在此基础上衍生出其他模块,例 ...

  5. ASP.NET Core中显示自定义错误页面-增强版

    之前的博文 ASP.NET Core中显示自定义错误页面 中的方法是在项目中硬编码实现的,当有多个项目时,就会造成不同项目之间的重复代码,不可取. 在这篇博文中改用middleware实现,并且放在独 ...

  6. javaweb学习6——自定义标签

    声明:本文只是自学过程中,记录自己不会的知识点的摘要,如果想详细学习JavaWeb,请到孤傲苍狼博客学习,JavaWeb学习点此跳转 本文链接:https://www.cnblogs.com/xdp- ...

  7. ASP.NET Core 配置 - 创建自定义配置提供程序

    ASP.NET Core 配置 - 创建自定义配置提供程序 在本文中,我们将创建一个自定义配置提供程序,从数据库读取我们的配置.我们已经了解了默认配置提供程序的工作方式,现在我们将实现我们自己的自定义 ...

  8. 在ASP.NET Core中创建自定义端点可视化图

    在上篇文章中,我为构建自定义端点可视化图奠定了基础,正如我在第一篇文章中展示的那样.该图显示了端点路由的不同部分:文字值,参数,动词约束和产生结果的端点: 在本文中,我将展示如何通过创建一个自定义的D ...

  9. ASP.NET Core 使用 JWT 自定义角色/策略授权需要实现的接口

    目录 ① 存储角色/用户所能访问的 API ② 实现 IAuthorizationRequirement 接口 ③ 实现 TokenValidationParameters ④ 生成 Token ⑤ ...

最新文章

  1. Vue 中 CSS 动画原理
  2. Michael Jordan:人工智能研究的目标变了,不再是构建单个智能
  3. winrar压缩指定目录
  4. 《深入理解 Java 内存模型》读书笔记(上)(干货,万字长文)
  5. 寻找数组中最小的k个数(快排和堆排)
  6. 初识消息队列/RabbitMQ详解
  7. IDEA2020的中文插件_IDEA2020个性化设置(装逼且实用)
  8. 网络游戏简易分区服务器架构详解
  9. 为了在简历上写掌握【Java集合】,做了万字总结
  10. 谷歌再推AI开源平台AI·ON,你有机会参与Bengio的项目了
  11. QAction QActionGroup QMenu 使用方法
  12. Failed to decrypt protected XML node DTS:Property with error 0x8009000B 错误的解决
  13. thinkpad选择启动项_ThinkPad如何设置光驱引导启动
  14. python基础教程-第1章节 基础知识
  15. Angular + PrimeNG 安装配置
  16. 不会编程,别着急!免编程工具助你快速开发App
  17. 开源资产管理系统SNIPE-IT搭建
  18. 元引擎视频制作工具 一键生成原创视频软件
  19. 头哥实践教学平台 CC++程序设计(计算机程序设计)基本输入输出 第2关:整数四则运算表达式的输出格式控制
  20. 计算机毕业设计Python+uniapp基于微信小程序某企业考勤系统(小程序+源码+LW)

热门文章

  1. python抓取数据包_利用python-pypcap抓取带VLAN标签的数据包方法
  2. 用Trivy扫描容器镜像
  3. 配置OpenShift 4 单机版环境 - CodeReady Container
  4. 使用Jenkins在Azure Web App上进行ASP.NET Core应用程序的持续集成和部署(CI/CD)–第4天
  5. python语言goto_如何在 Python 中实现 goto 语句
  6. python测试代码怎么写_python unittest编写测试代码
  7. 东北大学计算机 大一物理考试题,2010-2011东北大学物理考试题及答案
  8. 去掉chorme浏览器自动补全时input框的背景样式
  9. 外架小横杆外露长度规范要求_电缆制作通用工艺规范
  10. python django开发工具_Python和Django web开发工具pycharm介绍