前言:
我們經(jīng)常聽(tīng)到“秒修復(fù)秒上線”,覺(jué)得很厲害的樣子。
其實(shí)不然,這只是一個(gè)調(diào)侃而已,出現(xiàn)問(wèn)題的方式很多(邏輯漏洞、代碼異常、操作方式不正確等)。
我們今天來(lái)說(shuō)代碼異常問(wèn)題怎么快速定位,減少不必要的時(shí)間浪費(fèi)。
這就是今天的主題“添加全局異常處理機(jī)制”捕捉異常存儲(chǔ)到數(shù)據(jù)庫(kù)(mongodb、SqlServer、MySQL等)。
PS:輸出txt的話不怎么友好,不是所有人都能登錄服務(wù)器的。
異常是一種運(yùn)行時(shí)錯(cuò)誤,當(dāng)異常沒(méi)有得到適當(dāng)?shù)奶幚?,很可能?huì)導(dǎo)致你的程序意外終止。
1、創(chuàng)建項(xiàng)目
我們創(chuàng)建一個(gè)ASP.NET Core Web API
項(xiàng)目,選擇.NET Core3.1
。
2、創(chuàng)建全局異常過(guò)濾器
在控制器里面創(chuàng)建一個(gè)異常過(guò)濾器,命名為ExceptionFilter.cs
,過(guò)濾器繼承IExceptionFilter
接口。
注意,我這里入庫(kù)用的是efcore+sqlserver
,大家可以根據(jù)自己實(shí)際情況使用ORM
和數(shù)據(jù)庫(kù)方式。
異常過(guò)濾器,顧名思義,就是當(dāng)程序發(fā)生異常時(shí)所使用的過(guò)濾器。用于在系統(tǒng)出現(xiàn)未捕獲異常時(shí)的處理。
實(shí)現(xiàn)一個(gè)自定義異常過(guò)濾器,自定義一個(gè)全局異常過(guò)濾器需要實(shí)現(xiàn)IExceptionFilter
接口。
IExceptionFilter
接口會(huì)要求實(shí)現(xiàn)OnException
方法,當(dāng)系統(tǒng)發(fā)生未捕獲異常時(shí)就會(huì)觸發(fā)這個(gè)方法。
OnException方法有一個(gè)ExceptionContext
異常上下文,其中包含了具體的異常信息,HttpContext
及mvc
路由信息。
系統(tǒng)一旦出現(xiàn)未捕獲異常后,比較常見(jiàn)的做法就是使用日志工具,將異常的詳細(xì)信息記錄下來(lái),方便修正調(diào)試。
下面是日志記錄的實(shí)現(xiàn):
using Microsoft.AspNetCore.Mvc.Filters; using System; namespace Log4NetWebAPI.Controllers { public class ExceptionFilter: IExceptionFilter { //全局異常處理機(jī)制 public void OnException(ExceptionContext context) { Exception ex = context.Exception; //錯(cuò)誤所在的控制器方法名稱 var DisplayName = context.ActionDescriptor.DisplayName; #region 錯(cuò)誤所在的行號(hào)行號(hào) ////行號(hào)前的名稱有的是中文,有的是英文,注意甄別 //var aaa = ex.StackTrace.Substring(ex.StackTrace.IndexOf("行號(hào)"), ex.StackTrace.Length - ex.StackTrace.IndexOf("行號(hào)")); //var ccc = ex.StackTrace.Substring(ex.StackTrace.IndexOf("line"), ex.StackTrace.Length - ex.StackTrace.IndexOf("line")); var lineNumber = 0; const string lineSearch = ":line "; var index = ex.StackTrace.LastIndexOf(lineSearch); if (index != -1) { var lineNumberText = ex.StackTrace.Substring(index + lineSearch.Length); var lineNumberStr = lineNumberText.Substring(0, lineNumberText.IndexOf(" ")); if (int.TryParse(lineNumberStr, out lineNumber)) { } } #endregion #region 寫入信息到【數(shù)據(jù)庫(kù)】,這里自行入庫(kù)即可(mongodb、SqlServer、MySQL等),我以SqlServer為例 errorLog md = new errorLog(); md.logTime = DateTime.Now; md.logType = "error"; md.displayName = DisplayName; //錯(cuò)誤位置 md.lineNumber = lineNumber; //錯(cuò)誤行號(hào) md.message = ex.Message; //錯(cuò)誤信息 md.messagedetails = ex.ToString(); //錯(cuò)誤詳情 md.createTime= DateTime.Now; using (var ctx = new dbContext()) { ctx.Add(md); ctx.SaveChanges(); } #endregion //下面是你的返回頁(yè)面顯示的內(nèi)容提示,以下省略 } } }
3、依賴注入全局異常處理機(jī)制
在Startup.cs
里面把我們剛剛創(chuàng)建的全局異常處理機(jī)制注入到中間件來(lái),放到ConfigureServices
方法里面。
//添加全局異常處理機(jī)制
services.AddMvc(option => {
option.Filters.Add<ExceptionFilter>();
});
// This method gets called by the runtime. Use this method to add services to the container. public void ConfigureServices(IServiceCollection services) { services.AddControllers(); //添加全局異常處理機(jī)制 services.AddMvc(option => { option.Filters.Add<ExceptionFilter>(); }); }
4、測(cè)試全局異常處理機(jī)制
“將string
字符串轉(zhuǎn)換為int類型”,如下代碼,肯定是報(bào)錯(cuò)的,我們來(lái)捕捉一下錯(cuò)誤信息入庫(kù)。
var numberNo = "我是序列號(hào)";
//這里是轉(zhuǎn)換為int類型,然后數(shù)據(jù)源是string,肯定報(bào)錯(cuò),然后我們?nèi)植蹲?br /> //注意,我們這里沒(méi)有寫try catch
var number = Convert.ToInt32(numberNo);
運(yùn)行項(xiàng)目后我們查詢一下數(shù)據(jù)庫(kù),發(fā)現(xiàn)捕捉到錯(cuò)誤信息了,包括
錯(cuò)誤時(shí)間:2021-09-28 15:12:58.307
日志類型:error
錯(cuò)誤的方法位置:Log4NetWebAPI.Controllers.WeatherForecastController.Get (Log4NetWebAPI),
錯(cuò)誤的行號(hào):36行,
錯(cuò)誤的信息:Input string was not in a correct format.(輸入字符串的格式不正確。)
下面是控制器方法截圖
到此這篇關(guān)于ASP.NET Core
中間件的使用之全局異常處理機(jī)制的文章就介紹到這了,更多相關(guān)ASP.NET Core
全局異常處理機(jī)制內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://www.cnblogs.com/xiongze520/p/15348130.html