国产99久久精品_欧美日本韩国一区二区_激情小说综合网_欧美一级二级视频_午夜av电影_日本久久精品视频

最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
當前位置: 首頁 - 科技 - 知識百科 - 正文

ASP.NET CORE學習教程之自定義異常處理詳解

來源:懂視網 責編:小采 時間:2020-11-27 22:34:40
文檔

ASP.NET CORE學習教程之自定義異常處理詳解

ASP.NET CORE學習教程之自定義異常處理詳解:為什么異常處理選擇中間件? 傳統的ASP.NET可以采用異常過濾器的方式處理異常,在ASP.NET CORE中,是以多個中間件連接而成的管道形式處理請求的,不過常用的五大過濾器得以保留,同樣可以采用異常過濾器處理異常,但是異常過濾器不能處理MVC中間件以外的異
推薦度:
導讀ASP.NET CORE學習教程之自定義異常處理詳解:為什么異常處理選擇中間件? 傳統的ASP.NET可以采用異常過濾器的方式處理異常,在ASP.NET CORE中,是以多個中間件連接而成的管道形式處理請求的,不過常用的五大過濾器得以保留,同樣可以采用異常過濾器處理異常,但是異常過濾器不能處理MVC中間件以外的異

為什么異常處理選擇中間件?

傳統的ASP.NET可以采用異常過濾器的方式處理異常,在ASP.NET CORE中,是以多個中間件連接而成的管道形式處理請求的,不過常用的五大過濾器得以保留,同樣可以采用異常過濾器處理異常,但是異常過濾器不能處理MVC中間件以外的異常,為了全局統一考慮,采用中間件處理異常更為合適

為什么選擇自定義異常中間件?

先來看看ASP.NET CORE 內置的三個異常處理中間件 DeveloperExceptionPageMiddleware, ExceptionHandlerMiddleware,StatusCodePagesMiddleware

1.DeveloperExceptionPageMiddleware

能給出詳細的請求/返回/錯誤信息,因為包含敏感信息,所以僅適合開發環境

2.ExceptionHandlerMiddleware (蔣神博客:https://www.gxlcms.com/article/153926.htm)

僅處理500錯誤

3.StatusCodePagesMiddleware (蔣神博客:https://www.gxlcms.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請求,所以處理邏輯一定要精簡,防止發生不必要的性能問題

總結

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文檔

ASP.NET CORE學習教程之自定義異常處理詳解

ASP.NET CORE學習教程之自定義異常處理詳解:為什么異常處理選擇中間件? 傳統的ASP.NET可以采用異常過濾器的方式處理異常,在ASP.NET CORE中,是以多個中間件連接而成的管道形式處理請求的,不過常用的五大過濾器得以保留,同樣可以采用異常過濾器處理異常,但是異常過濾器不能處理MVC中間件以外的異
推薦度:
  • 熱門焦點

最新推薦

猜你喜歡

熱門推薦

專題
Top
主站蜘蛛池模板: 自拍 欧美 日韩 | 国产亚洲精品sese在线播放 | 国产精品va一区二区三区 | 国产欧美一区二区三区在线 | 久久99精品一区二区三区 | 国产精品永久免费自在线观看 | 在线视频欧美日韩 | 国产69精品久久久久777 | 欧美视频一区二区三区在线观看 | 日韩亚洲欧美日本精品va | 亚洲首页在线观看 | 欧美高清视频在线 | 国产日韩欧美中文字幕 | 欧美日韩亚洲高清不卡一区二区三区 | 久久一区二区三区免费 | 欧美成人高清视频 | 99999久久久久久亚洲 | 国产亚洲精品片a77777 | 久久99精品国产麻豆宅宅 | 日韩午夜电影 | 日b毛片 | 亚洲精品影院久久久久久 | 国内精品久久久久久久97牛牛 | 国产最新在线视频 | 国产毛片一区二区三区精品 | 久久亚洲精选 | 国产成人精品一区二区 | 人善交zzzxxx另类 | 日韩黄页| 亚洲成人精品久久 | 精品国产欧美一区二区五十路 | 精品在线看| 日本免费一区二区三区 | 免费在线观看亚洲 | 最新国产精品视频免费看 | 国产精品乱 | 日韩在线免费 | 黄色视频毛片 | 亚洲欧美国产高清va在线播放 | 亚洲日韩在线观看 | 日韩在线视频网 |