说一说MVC的CustomHandlerErrorAttribute(五)

九月第一篇,呵呵

前言:最近刚入职了一家公司,上司让我维护一个项目,我接手了看了一下项目,try catch 严重影响我的视觉,我直接通过vs插件将其try catch全部替换掉占位符,呵呵。

  所以我特此写了这篇文章...

正本:有可能你在搞开发的时候在每个模块中都要try catch,这样不仅不优雅也非常累,看了这篇文章,你会有所收获,你将会从50行代码收缩成30行.当然这取决于你的逻辑。

在CustomHandlerErrorAttribute中要继承HandleErrorAttribute

HandleErrorAttribute.cs

#region 程序集 System.Web.Mvc, Version=5.2.4.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35
#endregion

namespace System.Web.Mvc
{
    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = true)]
    public class HandleErrorAttribute : FilterAttribute, IExceptionFilter
    {
        public HandleErrorAttribute();
        public Type ExceptionType { get; set; }
        public string Master { get; set; }
        public override object TypeId { get; }
        public string View { get; set; }
        public virtual void OnException(ExceptionContext filterContext);
    }
}

人们常说虚方法都是可以重写的,也就是说你可以去扩展它.看代码CustomHandlerErrorAttribute.cs

 

 为什么我在这里截图呢,我是害怕别人复制黏贴,当然这不是我小气,因为很多人百度 都太懒了,也希望你们尊重博主的劳动成功

这样只要程序抛出异常,或者错误就换到自定义错误过滤器,这种形式也可以叫做短路器,当然不只是包括action的行为,如果是ajax请求报错,我们直接返回json,例如服务器崩掉。我们看一下测试过程。为了测试方便我直接把IIS启动起来了。

我们先看一下最简单的出错:

public ActionResult ExceptionUnCatch()
{
  throw new Exception("ExceptionCatch");
  return View();
}

 

@{
    ViewBag.Title = "ExceptionUnCatch";
}

<h2>ExceptionUnCatch</h2>

 

结果

经过测试也是成功跳入我的自定义错误过滤器,通过给action添加特性,才会使ASP.NET MVC 的生命周期中包括了我的过滤器。但是我们看另一种情况.我们加上try catch 呢?

 public ActionResult ExceptionCatch()
        {
            try
            {
                throw new Exception("ExceptionCatch...");
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
            }
            return View();
        }

视图方面:

@{
    ViewBag.Title = "ExceptionCatch";
}

<h2>ExceptionCatch</h2>

 可以发现错误被catch掉了,那如果我们在页面中抛出了异常呢.我们不去说view中会出什么异常,直接在代码中添加代码块<h1>@{throw new Exception("Exception View");}</h1>,经过我的测试,它也是成功的返回错误信息。

自习看过我的代码你会发现我在过滤器中添加了一个ajax请求的异常处理,我们简单的测试一下.在视图上定义了一个action  里面指向参数   我并没有那些控制器或action!!<h1>@Html.ActionLink("asdsa","asdsa")</h1>

结果   

 

最后在FilterConfig.cs中对所有action生效

  public static void RegisterGlobalFilters(GlobalFilterCollection filters)
        {
            filters.Add(new HandleErrorAttribute());
            //filters.Add(new CustomAuthorizeAttribute());//3 全局注册,全部的控制器/action都生效
            filters.Add(new CustomHandlerErrorAttribute());//
        }

 

 但是处理不到的情况就有了,例如在控制器构造函数中抛出异常或者说访问的url不存在

 

总来来说是因为它们是管道级别错误,不属于acton行为级别 ,这种情况我们只能借助配置项来对错误进行封装

在system.web中添加

 <customErrors mode="On" defaultRedirect="~/Demo.html">
      <error statusCode="404" redirect="~/Demo.html"/>
    </customErrors>

另一种方式可以在global.asax.cs中添加Application.error或许你应该懂了吧?

protected void Application_Error(object sender,EventArgs args)
{
     var errorDetail = Server.GetLastError();
     Context.Response.Write(errorDetail);
     Server.ClearError();
}

总结:使用自定义错误过滤器配合全局错误就可以达到非常好的错误处理。

posted @ 2018-09-01 16:26  ZaraNet  阅读(375)  评论(0编辑  收藏  举报