博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

ASP.NET中Application_Error对Exception的集中处理

Posted on 2011-10-16 15:18  随它去吧  阅读(5340)  评论(5编辑  收藏  举报

ASP.NET中对Exception的统一集中处理常见的有两种方式:

1、通过配置Web.config的customErrors节点,配置defaultRedirect属性来实现所有“未处理的异常”均跳转向同一页面

2、通过Global.asax中的Application_Error写代码实现自己想要的操作

但在实际使用当中,往往会出现一些问题,比如Application_Error中的代码未被执行,若使用customErrors那么如何传递错误信息给目标页面等等。

通过我的摸索,总结出一些经验,记录如下:

1、不想通过配置Web.config中customErrors的方法进行页面跳转,而是希望在Application_Error中写代码进行一些处理然后用Response.Redirect方法跳转到指定页面,这种理论上很不错的方法实际上行不通!Application_Error中写Response.Redirect来跳转页面不会执行!

2、很多人不想使用配置Web.config实现出错自动跳转的原因无非是:跳转到了一个“死”链接,无法传递当前的错误信息过去,也就没办法进行相应的处理!其实这是一个误解,任何一个aspx网页都可以使用Server.GetLastError()获得最后一个异常,也可以使用this.Context.AllErrors获得全部的未处理异常,你配置的页面当然也可以,只要他是一个aspx页面就行。

//获得最近一个Exception
Exception ex = this.Context.Server.GetLastError();
//获得所有未处理的Exception集
Exception[] errors = this.Context.AllErrors;

//然后你可以对他们进行处理了

3、网上有人推荐使用Web.config和Application_Error结合的方式来处理异常可以吗?当然可以!但个人感觉这不是个好主意,因为没啥必要。

我们看一下他们会怎么干:

Web.config

<customErrors mode="On" defaultRedirect="Error.aspx">

Global.asax

        protected void Application_Error(object sender, EventArgs e)
        {
            //获得最后一个Exception
            Exception ex = this.Context.Server.GetLastError();
            //然后进行一系列处理,写文件日志,写数据库都行

            //清除掉Exception,OK啦
            this.Context.Server.ClearError();
        }

Error.aspx

protected void Page_Load(object sender, EventArgs e)
{
        //输出期望的错误页面
}

这样看起来没问题吧,其实问题大了!你会发现它根本就不往Error.aspx页面跳转!为什么?因为Application_Error的处理优先级高于Web.config里面配置的自定义错误处理页面,你在Application_Error里面已经把所有该做的都做了,然后还把Error给Clear掉了,此时已经不会再触发Web.config里面配置的自动跳转了!

有人会说了,那我在Application_Error里面处理完之后不执行那个ClearError(),他不就跳转过去了吗?没错!这样一定是可以跳转的,而且你甚至可以在Error.aspx里面把这个错误再取出来一次,然后根据错误类型给用户一个“友好的”错误提示,最后别忘了再执行ClearError。这样是可以达到目的的,可是我真的觉得没必要,为什么没必要,看下面。

4、之所以有人采取3中的做法,无非是考虑到既想进行相应的错误捕捉处理,又想给用户一个“友好的”自定义界面。要达到这样的效果并不一定非得把两个结合起来,使用任意一种方法都能做到:

A、先看使用Web.config配置的方法:

Web.config

<customErrors mode="On" defaultRedirect="Error.aspx">

Error.aspx

protected void Page_Load(object sender, EventArgs e)
{
//获得最近一个Exception
Exception ex = this.Context.Server.GetLastError();

//然后你可以对他们进行处理了
}

B、使用Application_Error处理

Global.asax

        protected void Application_Error(object sender, EventArgs e)
        {
            //获取Exception
            Exception ex = this.Context.Server.GetLastError();
            //处理Exception

            //清除当前的输出
            this.Context.Response.Clear();
            //转向执行你希望展示给用户看的错误提示页面样子(此时网址依然是出错的那个页面,但是展示的内容就完全是你自己指定的页面了)
            this.Context.Server.Transfer("/Error.aspx");
       //如果非要跳转到目标页面(即地址栏也改变成Error.aspx的话)也不是不行,可以变通处理
       this.Context.Response.Write("<script>top.location.href='/Error.aspx';</script>");
        }