zoukankan      html  css  js  c++  java
  • MVC错误处理(一、二)

    mvc中经常报的错误:

    “/”应用程序中的服务器错误。


    无法找到资源。

    说明: HTTP 404。您正在查找的资源(或者它的一个依赖项)可能已被移除,或其名称已更改,或暂时不可用。请检查以下 URL 并确保其拼写正确。 

    请求的 URL: /sdf


    版本信息: Microsoft .NET Framework 版本:4.0.30319; ASP.NET 版本:4.0.30319.1 

    --------------------------------------------------------------------------------------------------------------

    MVC中,有一个Filter可以捕捉错误,但是它的用法是利用Attribute来实现的,而且只能加在Controller和Action上,所以不能捕捉别出的错误

    其实理论上所有的错误肯定产生于Controller中,但有2种情况下,就不会被捕捉了

    1、页面不存在的时候,找不到对应的Controller,那没有任何Controller被执行,所以自然也不会捕捉到错误了

    2、在 IAuthorizationFilter 下发生错误的时候,错误捕捉代码在IExceptionFilter中,而IAuthorizationFilter的优先权高于IExceptionFilter,所以也就捕捉不到了

     protected void Application_Error(object sender, EventArgs e)  

    复制代码
            {  
                Exception exception = Server.GetLastError();  
                Response.Clear();  
                HttpException httpException 
    = exception as HttpException;  
                RouteData routeData 
    = new RouteData();  
                routeData.Values.Add(
    "controller""Error");  
                
    if (httpException == null)  
                {  
                    routeData.Values.Add(
    "action""Index");  
                }  
                
    else //It's an Http Exception, Let's handle it.  
                {  
                    
    switch (httpException.GetHttpCode())  
                    {  
                        
    case 404:  
                            
    // Page not found.  
                            routeData.Values.Add("action""HttpError404");  
                            
    break;  
                        
    case 500:  
                            
    // Server error.  
                            routeData.Values.Add("action""HttpError500");  
                            
    break;  
                        
    // Here you can handle Views to other error codes.  
                        
    // I choose a General error template    
                        default:  
                            routeData.Values.Add(
    "action""General");  
                            
    break;  
                    }  
                }  
                
    // Pass exception details to the target error View.  
                routeData.Values.Add("error", exception.Message);  
                
    // Clear the error on server.  
                Server.ClearError();  
                
    // Call target Controller and pass the routeData.  
                IController errorController = new WEB.Controllers.ErrorController();  
                errorController.Execute(
    new RequestContext(  
               
    new HttpContextWrapper(Context), routeData));  
            }  
    复制代码
     

    把这段代码放到 Global.asax 中,并且新建一个 Controller 叫做 Error

     namespace MVC.Controllers  

    复制代码
    {  
        public class ErrorController : Controller  
        {  
            
    public ActionResult Index(string error)  
            {  
                ViewData[
    "Title"= "WebSite 网站内部错误";  
                ViewData[
    "Description"= error;  
                
    return View("Index");  
            }  
            
    public ActionResult HttpError404(string error)  
            {  
                ViewData[
    "Title"= "HTTP 404- 无法找到文件";  
                ViewData[
    "Description"= error;  
                
    return View("Index");  
            }  
            
    public ActionResult HttpError500(string error)  
            {  
                ViewData[
    "Title"= "HTTP 500 - 内部服务器错误";  
                ViewData[
    "Description"= error;  
                
    return View("Index");  
            }  
            
    public ActionResult General(string error)  
            {  
                ViewData[
    "Title"= "HTTP 发生错误";  
                ViewData[
    "Description"= error;  
                
    return View("Index");  
            }  
        }  
    }
    复制代码
     
    这样,就可以捕捉所有错误了。

    但其实,这样也不是完美的,因为如果你参考了我第一个问题中,在IIS6下不修改IIS设置,运行了MVC,那当后缀名不是.aspx的时候,错误不会被捕捉

    因为这时候输入的地址根本没有交给网站来处理,IIS直接抛出了错误,因为IIS认为这个后缀名不是你所能执行的.

    ------------------------------------------------------------------------------------------------------------------------------------------------

    很多网站如果由于用户输入错了地址,出现了如下图的网页

    又或者网站的bug导致的应用程序异常,搞出来个满屏的红黄黑,

    出现类似情况一定让用户大跌眼镜,个人认为,http错误与应用程序异常的处理方式应该是我们所需关注的问题。

    解决方案

    1.定义1个枚举类型用来存储http错误码,与应用程序异常错误

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    public enum DictSystemErrorType
    {
        /// <summary>
        /// 系统错误
        /// </summary>
        SystemError = 1,
        /// <summary>
        /// 系统异常
        /// </summary>
        SystemException = 2,
        /// <summary>
        /// 404错误
        /// </summary>
        Http404Error = 404,
        /// <summary>
        /// 500错误
        /// </summary>
        Http500Error = 500
    }
     

    2.定义SystemErrorCollection静态类用来管理错误提示信息

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    public static class SystemErrorCollection
    {
        private static readonly IDictionary<int, string> SystemMsg = new Dictionary<int, string>
                                                               {
                                                                   {1,"系统错误,请联系管理员!"},
                                                                   {2,"系统出现异常,请联系管理员!"},
                                                                   {404,"404错误,Really very sorry,The page not found!"},
                                                                   {500,"500错误,Internal Server Error!"},
                                                               };
        /// <summary>
        /// 获取错误提示
        /// </summary>
        /// <param name="errCode"></param>
        /// <returns></returns>
        public static string GetSystemErrorMsg(int  errCode)
        {
            return SystemMsg.SingleOrDefault(p => p.Key == errCode).Value;
        }
    }

    3.mvc下Global.asax文件和webForm下的一样,都继承自System.Web.HttpApplication,

     他们都包含Application_Error事件(当应用程序中遇到一个未处理的异常时,该事件被触发。

     定义Application_Error事件处理错误与异常

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    protected void Application_Error(object sender, EventArgs e)
    {
        Exception exception = Server.GetLastError();
        Response.Clear();
     
        var httpException = exception as HttpException;
        int errorCode = httpException == null ? (int)DictSystemErrorType.SystemException : httpException.GetHttpCode();
     
        //记录log  ...
        //发送错误邮件给网站管理人员  ...
     
        var routeData = new RouteData();
        routeData.Values.Add("controller", "Error");
        routeData.Values.Add("action", "error");
        routeData.Values.Add("errorCode", errorCode);
        Server.ClearError();
     
        IController errorController = new ErrorController();
        errorController.Execute(new RequestContext(new HttpContextWrapper(this.Context), routeData));
    }

    4.添加ErrorController与Error Action

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    public class ErrorController : Controller
    {
        //
        // GET: /Error/
     
        public ActionResult error()
        {
            int errorCode = (int)(this.RouteData.Values["errorCode"] ?? DictSystemErrorType.SystemError);
            ViewData["errorMsg"] = SystemErrorCollection.GetSystemErrorMsg(errorCode);
            return View();
        }
    }

     杂谈

    为方便管理错误码与提示信息,定义了枚举类型与一个IDictionary字典。

    然而同时维护这两个东西着实有些不变,还好可以通过反射取得枚举的提示信息

     不过最好把错误提示信息对应错误码持久化到数据库或者xml文件中,然后将其缓存起来。

    如此可随时更新错误信息,无需修改程序。

    IController是很简单的,它主要的用途在于提供了关于路由的工具来找到控制器并调用执行(Execute)

    Controller的HandleUnknownAction:控制器找不到相关的Action将会呼叫 HandleUnknownAction

    另外值得注意的是:

    mvc下如果你的某个Controller或者自定义基类的controller重写了HandleUnknownAction方法,

    那么出现http404错误的话,该Controller执行完HandleUnknownAction,将不会再执行Application_Error!

    -----------------------------------------------------------------------------------------------------------------------------------------------

  • 相关阅读:
    spin lock自旋锁 双链表操作(多线程安全)(Ring0)
    Interlocked单向链式栈
    自旋锁(Spin Lock)
    可等待定时器(获取系统时间)
    APC注入(Ring3)
    IOCP IO完成端口
    异步设备IO OVERLAPPED结构(设备内核对象 事件内核对象 可提醒IO)
    读写锁 SRWLOCK
    shell查看进程
    linux dmesg命令
  • 原文地址:https://www.cnblogs.com/8090sns/p/2936270.html
Copyright © 2011-2022 走看看