前言
在Spring MVC中,当一个请求发生异常(Controller抛出一个异常时), DispatcherServlet 采用委托的方式交给一个处理链来处理或者解析这个抛出的异常,这是在request和Servlet Container之间的一道屏障,所以我们可以在这里做一些处理工作,如转换异常,转换成友好的error page或者http 状态码等。
核心接口
这个处理机制在Spring是以HandlerExceptionResolver接口为核心的,该接口只有一个处理方法:
ModelAndView resolveException( HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex);
对于返回值 ModelAndView,有如下约定:
ModelAndView 指向一个页面 空的ModelAndView,表示异常已经在 HandlerExceptionResolver内部处理完成 null表示异常未处理,需要继续执行其它的HandlerExceptionResolver你可以通过声明多个HandlerExceptionResolver bean,并实现Ordered接口,来组成一个有顺序的HandlerExceptionResolver chain来处理异常。
Spring已经提供了以下几种实现:
SimpleMappingExceptionResolver 处理逻辑是根据Exception的class name映射成指定的error page。 DefaultHandlerExceptionResolver 是根据异常的类型转成http 状态码。 ResponseStatusExceptionResolver 是根据把异常和状态码通过@ResponseStatus绑定,当有异常抛出时,最终给客户端返回对应的状态码。 ExceptionHandlerExceptionResolver 是处理@ExceptionHandler的解析类,当有异常发生时,交给@ExceptionHandler方法去处理。处理逻辑
对ExceptionResolver的处理是在DispatcherServlet中进行的
DispatcherServlet有一个属性,这就表示前边提到的chain:
/** List of HandlerExceptionResolvers used by this servlet */ private List<HandlerExceptionResolver> handlerExceptionResolvers;
在DispatcherServlet初始化的时候同时对handlerExceptionResolvers进行了初始化
它从ApplicationContext中查询所有HandlerExceptionResolver bean,然后排序,上边这就是初始化工作.
在processHandlerException方法完成的对 HandlerExceptionResolver chain的调用,返回值不为null,即视作处理完成
// Check registered HandlerExceptionResolvers... ModelAndView exMv = null; for (HandlerExceptionResolver handlerExceptionResolver : this.handlerExceptionResolvers) { exMv = handlerExceptionResolver.resolveException(request, response, handler, ex); if (exMv != null) { break; } }
自定义异常处理器,可以实现HandlerExceptionResolver,也可以继承AbstractHandlerExceptionResolver类,实现doResolveException方法即可。
这里重点说下功能最为丰富的ExceptionHandlerExceptionResolver
通过@ExceptionHandler注解的方法,被视为异常处理方法,是通过ExceptionHandlerExceptionResolver来处理。该方法支持的参数类型有:
Exception Request/Response Session WebRequest or NativeRequest java.util.Locale java.io.InputStream / java.io.Reader java.io.OutputStream / java.io.Writer org.springframework.ui.Model支持的返回值类型有:
ModelAndView object (Servlet MVC or Portlet MVC). org.springframework.ui.Model java.util.Map object for exposing a model org.springframework.web.servlet.View. String value which is interpreted as view name. @ResponseBody 可以通过message converters 搭配内容协商来转换消息体. HttpEntity<?> or ResponseEntity<?> void 如果直接通过Response回写消息流,则该方法可以返回void