1. ExceptionHandler 注解
1. 简介
- ExceptionHandlerExceptionResolver
- 主要处理 Hanlder 中用 @ExceptionHandler 注解定义的方法
- @ExceptionHandler 注解定义的方法优先级问题:例如发生的是 NullPointerException,但是声明的异常有 RuntimeException 和 Exception,此时会根据异常的最近继承关系找到继承胜读最浅的那个 @ExceptionHandler 注解方法,即标记了 RuntimeException 的方法
- ExceptionHandlerMethodResolver 内部若找不到 @ExceptionHandler 注解的话,会找 @ControllerAdvice 中的 @ExceptionHandler 注解方法
2. 具体操作
1. 在 请求处理器类 SpringMVCTest 中加入方法
@ExceptionHandler({ArithmeticException.class})
public ModelAndView handleArithmeticException(Exception ex){
System.out.println("出异常了: " + ex);
ModelAndView mv = new ModelAndView("error");
mv.addObject("exception", ex);
return mv;
}
注意:
- 在 @ExceptionHandler 方法的入参中可以加入 Exception 类型的参数, 该参数即对应发生的异常对象
- @ExceptionHandler 方法的入参中不能传入 Map. 若希望把异常信息传导页面上, 需要使用 ModelAndView 作为返回值
- @ExceptionHandler 方法标记的异常有优先级的问题.
- @ControllerAdvice: 如果在当前 Handler 中找不到 @ExceptionHandler 方法来出来当前方法出现的异常,
则将去 @ControllerAdvice 标记的类中查找 @ExceptionHandler 标记的方法来处理异常.
2. 在 view 文件夹下创建视图 success.jsp 和 error.jsp
success.jsp
<h4>Success Page</h4>
error.jsp
<h4>Error Page</h4>
${exception}
3. 运行结果
前台
Error Page
java.lang.ArithmeticException: / by zero
后台
[FirstInterceptor] preHandle
----> 出异常了 java.lang.ArithmeticException: / by zero
[FirstInterceptor] afterCompletion
4. 分析优化
事实上在实际开发时,我们往往单独建立一个类来处理异常,并在类上添加 @ControllerAdvice 注解以及在方法上添加 @ExceptionHandler 注解
package www.xq.springmvc.test;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.servlet.ModelAndView;
@ControllerAdvice
public class SpringMVCTestExceptionHandler {
@ExceptionHandler({ArithmeticException.class})
public ModelAndView handleArithmeticException(Exception ex){
System.out.println("----> 出异常了 " + ex);
ModelAndView mv = new ModelAndView("error");
mv.addObject("exception", ex);
return mv;
}
}
下面三个异常解析器用法与上述大同小异。
2. ResponseStatusExceptionResolver
- 在异常及异常父类中找到 @ResponseStatus 注解,然后使用这个注解的属性进行处理。
- 定义一个 @ResponseStatus 注解修饰的异常类
@ResponseStatus(HttpStatus.UNAUTORIZED)
public class UnauthorizedException extends RuntimeException{}
3. DefaultHandlerExceptionResolver
- 对一些特殊异常进行处理,比如:
- NoSuchRequestHandlingMethodException
- HttpRequestMethodNotSupportedException
- HttpMediaTypeNotSupportedException
- HttpMediaTypeNotAcceptableException
- …
4. SimpleMappingExceptionResolver
- 如果希望对所有异常进行统一处理,可以使用 SimpleMappingExceptionResolver,它将异常类名映射为视图名,即发生异常时使用对应的视图报告异常
<!-- 配置使用 SimpleMappingExceptionResolver 来映射异常 -->
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<property name="exceptionAttribute" value="ex"></property>
<property name="exceptionMappings">
<props>
<prop key="java.lang.ArrayIndexOutOfBoundsException">error</prop>
</props>
</property>
</bean>