java web项目整体异常处理机制

java web项目整体异常处理机制  

实际的j2ee项目中,系统内部难免会出现一些异常,如果把异常放任不管直接打印到浏览器可能会让用户感觉莫名其妙,也有可能让某些用户找到破解系统的方法。

就拿spring+struts2+hibernate项目说明:通常一个页面请求到后台以后,首先是到action(也就是所谓mvccontroller),在action层会调用业务逻辑serviceservce层会调用持久层dao获取数据。最后执行结果会汇总到action,然后通过action控制转发到指定页面,执行流程如下图所示:

java <wbr>web项目整体异常处理机制


而这三层其实都有可能发生异常,比如dao层可能会有SQLExceptionservice可能会有NullPointExceptionaction可能会有IOException,一但发生异常并且程序员未做处理,那么该层不会再往下执行,而是向调用自己的方法抛出异常,如果daoserviceaction层都未处理异常的话,异常信息会抛到服务器,然后服务器会把异常直接打印到页面,结果就会如下图所示:

java <wbr>web项目整体异常处理机制

 其实这种错误对于客户来说毫无意义,因为他们通常是看不懂这是什么意思的。

刚学java的时候,我们处理异常通常两种方法:直接throws,放任不管;try...catch,在catch块中不作任何操作,或者仅仅printStackTrace()把异常打印到控制台。第一种方法最后就造就了上图的结果;而第二种方法更杯具:页面不报错,但是也不执行用户的请求,简单的说,其实这就是bug(委婉点:通常是这样)!

 

那么发生异常到底应该怎么办呢?我想在大家对java异常有一定了解以后,会知道:异常应该在action控制转发之前尽量处理,同时记录log日志,然后在页面以友好的错误提示告诉用户出错了。大家看下面的代码:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客
//
创建日志对象   
Log log LogFactory.getLog(this.getClass());   
  
//action
层执行数据添加操作   
public String save(){   
   try{   
         //
调用servicesave方法   
         service.save(obj);   
   }catch(Exception e){   
         log.error(...);   //
记录log日志   
      return "error"; 
到指定error页面   
   }   
   return "success";   
 

如果按照上面的方式处理异常以后,我们用户最后看到的页面可能就会是下面这种形式(我想这种错误提示应该稍微友好点了吧):


java <wbr>web项目整体异常处理机制

  

 

然后我们回到刚才处理异常的地方,如果大家积累了一些项目经验以后会发现使用上面那种处理异常的方式可能还不够灵活:

因为spring把大多数非运行时异常都转换成运行时异常(RuntimeException)最后导致程序员根本不知道什么地方应该进行try...catch操作

每个方法都重复写try...catch,而且catch块内的代码都很相似,这明显做了很多重复工作而且还很容易出错,同时也加大了单元测试的用例数(项目经理通常喜欢根据代码行来估算UT case

发生异常有很多种情况:可能有数据库增删改查错误,可能是文件读写错误,等等。用户觉得每次发生异常都是访问过程中产生错误,请重试的提示完全不能说明错误情况,他们希望让异常信息更详尽些,比如:在执行数据删除时发生错误,这样他们可以更准确地给维护人员提供bug信息。

如何解决上面的问题呢?我是这样做的:JDK异常或自定义异常+异常拦截器

struts2拦截器的作用在网上有很多资料,在此不再赘述,我的异常拦截器原理如下图所示:

java <wbr>web项目整体异常处理机制

 首先我的action类、service类和dao类如果有必要捕获异常,我都会try...catchcatch块内不记录log,通常是抛出一个新异常,并且注明错误信息:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客
//action
层执行数据添加操作   
public String save(){   
   try{   
         //
调用servicesave方法   
         service.save(obj);   
   }catch(Exception e){   
      //
你问我为什么抛出Runtime异常?因为我懒得在方法后写throws  xx   
      throw new RuntimeException("
添加数据时发生错误!",e);   
  }   
   return "success";   
 

 

Struts2自定义拦截器的操作:

struts2的配置文件下添加:

    <</span>interceptors>

        

         <</span>interceptor name="errorInterceptor" class="cn.itcast.elec.util.ErrorInterceptor" />

        

         <</span>interceptor-stack name="myErrorInterceptor">

                     <</span>interceptor-ref name="defaultStack" />

                     <</span>interceptor-ref name="errorInterceptor" />

         </</span>interceptor-stack>

       </</span>interceptors>

     

      <</span>default-interceptor-ref name="myErrorInterceptor"/>

 

然后在异常拦截器对异常进行处理,看下面的代码:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客
public String intercept(ActionInvocation actioninvocation) {   
  
        String result null; // Action
的返回值   
        try {   
            // 
运行被拦截的Action,期间如果发生异常会被catch   
            result actioninvocation.invoke();   
            return result;   
        catch (Exception e) {   
              
            String errorMsg "
未知错误!";   
            //
通过instanceof判断到底是什么异常类型   
            if (e instanceof BaseException) {   
                BaseException be (BaseException) e;   
                be.printStackTrace(); //
开发时打印异常信息,方便调试   
                if(be.getMessage()!=null||Constants.BLANK.equals(be.getMessage().trim())){   
                    //
获得错误信息   
                    errorMsg be.getMessage().trim();   
                }   
            else if(e instanceof RuntimeException){   
                //
未知的运行时异常   
                RuntimeException re (RuntimeException)e;   
                re.printStackTrace();   
            else{   
                //
未知的严重异常   
                e.printStackTrace();   
            }   
            //
把自定义错误信息   
            HttpServletRequest request (HttpServletRequest) actioninvocation   
                    .getInvocationContext().get(StrutsStatics.HTTP_REQUEST);   
               
              
            request.setAttribute("errorMsg", errorMsg);   
           
              
            Log log LogFactory   
                    .getLog(actioninvocation.getAction().getClass());   
            if (e.getCause() != null){   
                log.error(errorMsg, e);   
            }else{   
                log.error(errorMsg, e);   
            }   
  
            return "error";   
        }// ...end of catch   
     

 需要注意的是:在使用instanceof判断异常类型的时候一定要从子到父依次找,比如BaseException继承与RuntimeException,则必须首先判断是否是BaseException再判断是否是RuntimeException

 

 

最后在error JSP页面显示具体的错误消息即可:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客

   
   
       

对不起,系统发生了未知的错误

   
   
   
       

${requestScope.errorMsg}

   
   
   

 

    

对不起,系统发生了未知的错误

       

${requestScope.errorMsg}

   

 

以上方式可以拦截后台代码所有的异常,但如果出现数据库连接异常时不能被捕获的,大家可以使用struts2的全局异常处理机制来处理:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客
  
    /Web/common/page/error.jsp   
  
            
  
       
  

上面这是一个很简单的异常拦截器,大家可以使用自定义异常,那样会更灵活一些。

 

以上异常拦截器可以使用其它很多技术替换:比如spring aopservlet filter等,根据项目实际情况处理。

 

【补充】ajax也可以进行拦截,但是因为ajax属于异步操作,action通过response形式直接把数据返回给ajax回调函数,如果发生异常,ajax是不会执行页面跳转的,所以必须把错误信息返回给回调函数,我针对json数据的ajax是这样做的:


Java
代码 复制代码   收藏代码 java web项目整体异常处理机制 - fanxiaoqc@126 - Fay的博客
  
HttpServletResponse response (HttpServletResponse)actioninvocation.getInvocationContext().get(StrutsStatics.HTTP_RESPONSE);   
response.setCharacterEncoding(Constants.ENCODING_UTF8);   
  
PrintWriter out;   
try {   
    out response.getWriter();   
    Message msg new Message(errorMsg);   
    //
把异常信息转换成json格式返回给前台   
    out.print(JSONObject.fromObject(msg).toString());   
catch (IOException e1) {   
    throw e;   
}

   

1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值