您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關ASP.NET MVC錯誤處理的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
ASP.NET MVC的錯誤處理應考慮到這幾個方面:模型綁定期間發生的錯誤,未能路由到指定操作,針對控制器的錯誤處理。使用配置文件可以幫助我們處理異常,但是不夠靈活和全面;使用HandleErrorAttribute、自定義錯誤過濾器或重寫控制器OnException方法只能解決針對控制器的錯誤,無法解決模型綁定期間發生的錯誤,也無法處理404錯誤,即使將錯誤過濾器注冊為全局過濾器也是如此。有時候需要多種方法配合使用。
在捕獲錯誤的地方,可以將有用的信息記錄下來,便于我們查出引起問題的原因和糾正錯誤。
1啟用自定義錯誤
使用這種方式一定要注意將defaultRedirect設置為指定的錯誤頁面,防止黑客探測各種錯誤情形進而發現系統的額漏洞。
<system.web> <customErrors mode="On" defaultRedirect="/error/error2"> <error statusCode="404" redirect="/error/error1" /> </customErrors> <!--其他配置--> </system.web>
Mode:處理模式,有三種處理模式
On,啟用自定義處理功能,當錯誤發生時顯示自定義錯誤頁
Off,關閉自定義錯誤處理功能,當錯誤發生時顯示默認的錯誤頁。
RemoteOnly,啟用自定義錯誤處理功能,但只針對來自遠程機器的請求有效。
defaultRedirect:發生錯誤時,顯示指定錯誤頁。
<error>:根據狀態碼顯示指定的錯誤頁。mode必須為On或RemoteOnly模式,否則不會起作用。
注意:不論defaultRedirect和redirect都配置為指定的路徑,例如上述配置中控制器error,控制器操作為error1和error2,相應地錯誤頁為Error1.cshtml和Error2.cshtml。
2針對控制器的錯誤處理
2.1使用HandleErrorAttribute修飾控制器或操作。
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = true)] public class HandleErrorAttribute : FilterAttribute, IExceptionFilter { //獲取或設置異常的類型。 public Type ExceptionType { get; set; } //獲取或設置用于顯示異常信息的母版視圖 public string Master { get; set; } //獲取此特性的唯一標識符。 public override object TypeId { get; } //獲取或設置用于顯示異常信息的頁視圖。 public string View { get; set; } //在發生異常時調用。 //filterContext:操作篩選器上下文 public virtual void OnException(ExceptionContext filterContext); }
例:
當發生KeyNotFoundException類型的異常時,顯示KeyNotFind視圖
[HandleError(ExceptionType=typeof(KeyNotFoundException),View="KeyNotFound")] public ActionResult Index() { ...... }
還可以使用自定義的錯誤過濾器,并將其應用到控制器或操作上。
例:
public class CustomHandleError : HandleErrorAttribute { public override void OnException(ExceptionContext filterContext) { if (filterContext==null) base.OnException(filterContext); //記錄日志 LogError(filterContext.Exception); //判斷是否啟用了自定義錯誤 if (filterContext.HttpContext.IsCustomErrorEnabled) { //將錯誤設置為已處理 filterContext.ExceptionHandled = true; base.OnException(filterContext); } } }
可以設置全局過濾器,這樣對每一個控制器都起作用。
App_Start文件夾下FilterConfig.cs文件中設置全局錯誤過濾器,過濾器會按照他們注冊的順序執行。但可以通過Order屬性指定執行順序。
例:
public class FilterConfig { public static void RegisterGlobalFilters(GlobalFilterCollection filters) { filters.Add(new HandleErrorAttribute { ExceptionType = typeof(KeyNotFoundException), View = "KeyNotFound", Order = 2 }); filters.Add(new HandleErrorAttribute(),1); } }
將自定義錯誤過濾器設置為全局過濾器:
在App_Start文件夾下FilterConfig.cs文件中
例:
public class FilterConfig { public static void RegisterGlobalFilters(GlobalFilterCollection filters) { //其他過濾器 filters.Add(new CustomHandleError()); } }
2.2重寫控制器OnException方法
注意將錯誤設置為已處理,不然錯誤繼續拋出,但如果設置了全局錯誤過濾器,那么即使不標記為已處理,也不要緊,因為錯誤最終會被全局過濾器捕獲并處理。
例:
public class HomeController : Controller { //其他控制器操作 protected override void OnException(ExceptionContext filterContext) { if (filterContext==null) base.OnException(filterContext); //記錄日志 LogError(filterContext.Exception); //判斷是否啟用了自定義錯誤 if (filterContext.HttpContext.IsCustomErrorEnabled) { //將錯誤設置為已處理 filterContext.ExceptionHandled = true; //顯示錯誤頁 this.View("Error").ExecuteResult(this.ControllerContext); } } }
或者創建控制器基類
public class BaseController : Controller { protected override void OnException(ExceptionContext filterContext) { //錯誤日志記錄 } }
3全局錯誤處理
針對模型綁定或路由等過程中拋出的異常我們只能使用全局錯誤處理策略。
3.1 Global.asax中添加處理異常的代碼
例:
public class MvcApplication : System.Web.HttpApplication { protected void Application_Start() { AreaRegistration.RegisterAllAreas(); FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); RouteConfig.RegisterRoutes(RouteTable.Routes); BundleConfig.RegisterBundles(BundleTable.Bundles); } protected void Application_Error(object sender, EventArgs e) { var exception = Server.GetLastError(); if (exception == null) { return; } //異常發生記錄日志或發送郵件 //清除異常 Server.ClearError(); //重定向 Response.Redirect("home/index"); } }
3.2捕獲未匹配的路由
在路由注冊列表最底端注冊路由。
public class RouteConfig { public static void RegisterRoutes(RouteCollection routes) { //其他配置 routes.MapRoute( name: "MatchAll", url: "{*anyurl}", defaults: new { controller = "Error",action ="Missing" } ); } }
定義Error控制器及Missing操作
public class ErrorController : Controller { // GET: Error public ActionResult Missing() { HttpContext.Response.StatusCode = 404; //禁用IIS7默認的錯誤頁,這樣才能展示我們指定都的視圖 HttpContext.Response.TrySkipIisCustomErrors = true; //也可以在此處記錄日志信息 //要展示的信息 var model = ...... return View(model); } }
需要注意的是,這種方式不一定能處理所有未匹配的情形。
例如:http://localhost/mvcpointapp/home/index1,這個url請求說我home是存在,但是index1操作不存在,上面配置MatchAll路由無法匹配這個url。
可以匹配的情形如:http://localhost/mvcpointapp/v1/home/index/1,這個url能被上面配置的MatchAll路由匹配,所以可以顯示Missing視圖。
4實踐
4.1使用HandleErrorAttribute注意要對<system.web>的<customErrors>節進行設置 。
例如:
控制器為
public class HomeController : Controller { [HandleError(ExceptionType = typeof(KeyNotFoundException), View = "Error")] public ActionResult Index() { throw new KeyNotFoundException(); return View(); } //其他控制操作 }
<system.web>的<customErrors>節
<customErrors mode="On" defaultRedirect="Error/Error2"></customErrors>
Error.cshtml文件位于Views文件夾下的子文件夾Shared文件夾下
瀏覽器中輸入:http://localhost/mvcpointapp/
結果可以正常顯示Error.cshtml頁面,同時注意到雖然在customErrors 配置節中指定了defaultRedirect,但還是跳轉到Error.cshtml頁面。
將<customErrors>的mode設置為Off,則顯示經典錯誤頁。
4.2 Application_Error
代碼如3.1節所示,控制器如4.1所示,<system.web>的<customErrors>節為<customErrors mode="On" defaultRedirect="Error/Error2"></customErrors>
輸入:http://localhost/mvcpointapp/home/index,斷點調試,發現錯誤被HandleError攔截,Global.asax的Application_Error方法沒有執行。而當輸入:http://localhost/mvcpointapp/home/indexr,Application_Error執行了。
關閉<customErrors>配置節,而不注掉控制器上的HandleErrorAttribute特性,輸入:http://localhost/mvcpointapp/home/index,發現Application_Error執行了。
通過上述實踐,充分證明HandleErrorAttribute會攔截控制器內拋出的異常,而無法攔截無法找到資源這種異常。
4.3策略
一種常用的攔截錯誤信息、記錄錯誤日志與顯示自定義錯誤頁的策略為:
1)首先配置<system.web>的<customErrors>節,注意務必設置defaultRedirect;并且定義錯誤控制器及相應的操作和視圖。
2)定義基類控制器或自定義錯誤過濾器,記錄異常。對于自定義錯誤過濾器的情形一般將其注冊為全局過濾器。
3)在Global.asax中添加Application_Error方法攔截意想不到的異常并記錄異常。
關于“ASP.NET MVC錯誤處理的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。