Spring-MVC开发之全局异常捕获全面解读(转)

异常,异常。我们一定要捕获一切该死的异常,宁可错杀一千也不能放过一个!产品上线后的异常更要命,一定要屏蔽错误内容,以免暴露敏感信息!在用Spring MVC开发WEB应用时捕获全局异常的方法基本有两种:

  • WEB.XML,就是指定error-code和page到指定地址,这也是最传统和常见的做法
  • 用Spring的全局异常捕获功能,这种相对可操作性更强一些,可根据自己的需要做一后善后处理,比如日志记录等。

SO,本文列出Spring-MVC做WEB开发时常用全局异常捕获的几种解决方案抛砖引玉,互相没有依赖,每个都可单独使用!

1、定义服务器错误WEB.XML整合Spring MVC

web.xml:

 1 <error-page>
 2     <error-code>404</error-code>
 3     <location>/404</location>
 4 </error-page>
 5 <error-page>
 6     <error-code>500</error-code>
 7     <location>/500</location>
 8 </error-page>
 9 
10 <!-- 未捕获的错误,同样可指定其它异常类,或自定义异常类 -->
11 <error-page>
12     <exception-type>java.lang.Exception</exception-type>
13     <location>/uncaughtException</location>
14 </error-page>

applicationContext.xml:

1 <!-- 错误路径和错误页面,注意指定viewResolver -->
2 <mvc:view-controller path="/404" view-name="404"/>
3 <mvc:view-controller path="/500" view-name="500"/>
4 <mvc:view-controller path="/uncaughtException" view-name="uncaughtException"/>

 2、Spring全局异常,Controller增强方式( Advising Controllers)

异常抛出:

1 @Controller
2 public class MainController {
3     @ResponseBody
4     @RequestMapping("/")
5     public String main(){
6         throw new NullPointerException("NullPointerException Test!");
7     }
8 }

异常捕获:

 1 //注意使用注解@ControllerAdvice作用域是全局Controller范围,即必须与抛出异常的method在同一个controller
 2 //可应用到所有@RequestMapping类或方法上的@ExceptionHandler、@InitBinder、@ModelAttribute,在这里是@ExceptionHandler
 3 @ControllerAdvice
 4 public class AControllerAdvice {
 5     @ExceptionHandler(NullPointerException.class)
 6     @ResponseStatus(HttpStatus.BAD_REQUEST)
 7     @ResponseBody
 8     public String handleIOException(NullPointerException ex) {
 9         return ClassUtils.getShortName(ex.getClass()) + ex.getMessage();
10     }
11 }

为了确保@ResponseStatus标注的异常被Spring框架处理,可以这样编写全局异常处理类:

 1 @ControllerAdvice
 2 class GlobalDefaultExceptionHandler {
 3     public static final String DEFAULT_ERROR_VIEW = "error";
 4 
 5     @ExceptionHandler(value = Exception.class)
 6     public ModelAndView defaultErrorHandler(HttpServletRequest req, Exception e) throws Exception {
 7         // If the exception is annotated with @ResponseStatus rethrow it and let
 8         // the framework handle it - like the OrderNotFoundException example
 9         // at the start of this post.
10         // AnnotationUtils is a Spring Framework utility class.
11         if (AnnotationUtils.findAnnotation(e.getClass(), ResponseStatus.class) != null)
12             throw e;
13 
14         // Otherwise setup and send the user to a default error-view.
15         ModelAndView mav = new ModelAndView();
16         mav.addObject("exception", e);
17         mav.addObject("url", req.getRequestURL());
18         mav.setViewName(DEFAULT_ERROR_VIEW);
19         return mav;
20     }
21 }

3、Spirng全局异常,配置方式

异常抛出,同上!

异常捕获:

 1 <!-- 全局异常配置 -->
 2     <bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
 3         <property name="exceptionMappings">
 4             <props>
 5                 <prop key="java.lang.Exception">errors/500</prop>
 6                 <prop key="java.lang.Throwable">errors/500</prop>
 7             </props>
 8         </property>
 9         <property name="statusCodes">
10             <props>
11                 <prop key="errors/500">500</prop>
12             </props>
13         </property>
14         <!-- 设置日志输出级别,不定义则默认不输出警告等错误日志信息 -->
15         <property name="warnLogCategory" value="WARN"></property>
16         <!-- 默认错误页面,当找不到上面mappings中指定的异常对应视图时,使用本默认配置 -->
17         <property name="defaultErrorView" value="errors/500"></property>
18         <!-- 默认HTTP状态码 -->
19         <property name="defaultStatusCode" value="500"></property>
20     </bean>

对应500错误的view jsp页面:

 1 <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
 2 <%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
 3 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
 4 <html>
 5 <head>
 6 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 7 <title>500 Error</title>
 8 </head>
 9 <body>
10     <% Exception ex = (Exception)request.getAttribute("exception"); %>
11     <H2>Exception: <%= ex.getMessage()%></H2>
12     <P/>
13     <% ex.printStackTrace(new java.io.PrintWriter(out)); %>
14 </body>
15 </html>

4、Sping全局异常,自定义异常类和异常解析
自定义异常类:

 1 public class CustomException extends RuntimeException {
 2 
 3     public CustomException(){
 4         super();
 5     }
 6 
 7     public CustomException(String msg, Throwable cause){
 8         super(msg, cause);
 9         //Do something...
10     }
11 }

抛出异常:

1 @ResponseBody
2 @RequestMapping("/ce")
3 public String ce(CustomException e){
4     throw new CustomException("msg",e);
5 }

实现异常捕获接口HandlerExceptionResolver:

 1 public class CustomHandlerExceptionResolver implements HandlerExceptionResolver{
 2 
 3     @Override
 4     public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
 5         Map<String, Object> model = new HashMap<String, Object>();
 6         model.put("e", e);
 7         //这里可根据不同异常引起类做不同处理方式,本例做不同返回页面。
 8         String viewName = ClassUtils.getShortName(e.getClass());
 9         return new ModelAndView(viewName, model);
10     }
11 }

新的的HandlerExceptionResolver实现类只需在配置文件中定义即可,可以配置优先级。DispatcherServlet初始化HandlerExceptionResolver的时候会自动寻找容器中实现了HandlerExceptionResolver接口的类,然后添加进来。配置Spring支持异常捕获:

1 <bean class="cn.bg.controller.CustomHandlerExceptionResolver"/>

5、Errors and REST

使用Restful的Controller可以使用@ResponseBody处理错误,首先定义一个错误:

1 public class ErrorInfo {
2     public final String url;
3     public final String ex;
4 
5     public ErrorInfo(String url, Exception ex) {
6         this.url = url;
7         this.ex = ex.getLocalizedMessage();
8     }
9 }

通过一个@ResponseBody返回一个错误实例:

1 @ResponseStatus(HttpStatus.BAD_REQUEST)
2 @ExceptionHandler(MyBadDataException.class)
3 @ResponseBody ErrorInfo handleBadRequest(HttpServletRequest req, Exception ex) {
4     return new ErrorInfo(req.getRequestURL(), ex);
5 } 

 

6、参考:

原文:Spring-MVC开发之全局异常捕获全面解读

[Spring MVC] - 500/404错误处理

Exception Handling in Spring MVC

posted @ 2015-03-07 00:12  添哥  阅读(23747)  评论(0编辑  收藏  举报