ASP.NET CORE学习教程之自定义异常处理详解
 更新时间:2019年01月03日 08:35:40   作者:#山鸡  

这篇文章主要给大家介绍了关于ASP.NET CORE学习教程之自定义异常处理的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

为什么异常处理选择中间件?
传统的ASP.NET可以采用异常过滤器的方式处理异常,在ASP.NET CORE中,是以多个中间件连接而成的管道形式处理请求的,不过常用的五大过滤器得以保留,同样可以采用异常过滤器处理异常,但是异常过滤器不能处理MVC中间件以外的异常,为了全局统一考虑,采用中间件处理异常更为合适
为什么选择自定义异常中间件?
先来看看ASP.NET CORE 内置的三个异常处理中间件 DeveloperExceptionPageMiddleware, ExceptionHandlerMiddleware,StatusCodePagesMiddleware
1.DeveloperExceptionPageMiddleware

能给出详细的请求/返回/错误信息,因为包含敏感信息,所以仅适合开发环境
2.ExceptionHandlerMiddleware (蒋神博客:www.cnhackhy.com/article/153926.htm)
仅处理500错误
3.StatusCodePagesMiddleware (蒋神博客:www.cnhackhy.com/article/153931.htm)
能处理400-599之间的错误,但需要Response中不能包含内容(ContentLength=0 && ContentType=null,经实验不能响应mvc里未捕获异常)
由于ExceptionHandlerMiddleware和StatusCodePagesMiddleware的各自的限制条件,两者需要搭配使用。相比之下自定义中间件更加灵活,既能对各种错误状态进行统一处理,也能按照配置决定处理方式。
CustomExceptionMiddleWare
首先声明异常中间件的配置类

/// <summary>
/// 异常中间件配置对象
/// </summary>
public class CustomExceptionMiddleWareOption
{
public CustomExceptionMiddleWareOption(
CustomExceptionHandleType handleType = CustomExceptionHandleType.JsonHandle,
IList<PathString> jsonHandleUrlKeys = null,
string errorHandingPath = “”)
{
HandleType = handleType;
JsonHandleUrlKeys = jsonHandleUrlKeys;
ErrorHandingPath = errorHandingPath;
}

/// <summary>
/// 异常处理方式
/// </summary>
public CustomExceptionHandleType HandleType { get; set; }

/// <summary>
/// Json处理方式的Url关键字
/// <para>仅HandleType=Both时生效</para>
/// </summary>
public IList<PathString> JsonHandleUrlKeys { get; set; }

/// <summary>
/// 错误跳转页面
/// </summary>
public PathString ErrorHandingPath { get; set; }
}

/// <summary>
/// 错误处理方式
/// </summary>
public enum CustomExceptionHandleType
{
JsonHandle = 0, //Json形式处理
PageHandle = 1, //跳转网页处理
Both = 2 //根据Url关键字自动处理
}

声明异常中间件的成员

/// <summary>
/// 管道请求委托
/// </summary>
private RequestDelegate _next;

/// <summary>
/// 配置对象
/// </summary>
private CustomExceptionMiddleWareOption _option;

/// <summary>
/// 需要处理的状态码字典
/// </summary>
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);
else
await PageHandle(context, exception, _option.ErrorHandingPath);
}
}
}

/// <summary>
/// 统一格式响应类
/// </summary>
/// <param name=”ex”></param>
/// <returns></returns>
private ApiResponse GetApiResponse(Exception ex)
{
return new ApiResponse() { IsSuccess = false, Message = ex.Message };
}

/// <summary>
/// 处理方式:返回Json格式
/// </summary>
/// <param name=”context”></param>
/// <param name=”ex”></param>
/// <returns></returns>
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);
}

/// <summary>
/// 处理方式:跳转网页
/// </summary>
/// <param name=”context”></param>
/// <param name=”ex”></param>
/// <param name=”path”></param>
/// <returns></returns>
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<CustomExceptionMiddleWare>(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异常和错误处理(8)ASP.NET Core中实现全局异常拦截的完整步骤

core
自定义
异常

相关文章
Asp.net webForm设置允许表单提交Html的方法下面小编就为大家分享一篇Asp.net webForm设置允许表单提交Html的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 2017-12-12
ASP.NET实现读取Excel内容并在Web上显示这篇文章主要介绍了ASP.NET实现读取Excel内容并在Web上显示,很实用的一个技巧,需要的朋友可以参考下 2014-08-08
ASP.NET简化编辑界面解决思路及实现代码简化用户操作界面,添加功能一般没法简化,但是如果是在GridView做显示,编辑,更新与删除,会让用户在编辑,需要点击编辑铵钮,再进行编辑,或是取消编辑,下面是改进的方法做了演示动画,感兴趣的朋友可以了解下 2013-01-01
mvc file控件无刷新异步上传操作源码这篇文章主要为大家分享了mvc file控件无刷新异步上传操作源码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 2016-10-10
ASP.NET Core使用HostingStartup增强启动操作方法详解这篇文章主要介绍了ASP.NET Core使用HostingStartup增强启动操作,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下 2020-11-11
ASP.NET使用TreeView显示文件的方法这篇文章主要介绍了ASP.NET使用TreeView显示文件的方法,包括控件的使用、页面加载代码及节点事件,是WEB应用程序中非常实用的技巧,需要的朋友可以参考下 2014-09-09
moq 的常用使用方法(推荐)这篇文章主要介绍了moq 的常用使用方法的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下 2016-11-11
C# 没有动态的数组,可以用arraylist或list取代C#里没有动态的数组,只能用arraylist或list取代。 2009-06-06
详解Asp.Net Core 发布和部署( MacOS + Linux + Nginx )这篇文章主要介绍了详解Asp.Net Core 发布和部署( MacOS + Linux + Nginx ) ,具有一定的参考价值,有兴趣的可以了解一下。 2016-12-12
深入Lumisoft.NET组件POP3邮件接收与删除操作的使用详解本篇文章对Lumisoft.NET组件POP3邮件接收与删除操作的使用进行了详细的介绍。需要的朋友参考下 2013-05-05

最新评论

声明:本站(华域联盟www.cnhackhy.com)所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。