SPRing MVC也可以使用拦截器对请求进行拦截处理,用户可以自定义拦截器来实现特定的功能,自定义的拦截器必须实现HandlerInterceptor接口:
preHandle():这个方法在业务处理器处理请求之前被调用,在该方法中对用户请求 request 进行处理。如果程序员决定该拦截器对请求进行拦截处理后还要调用其他的拦截器,或者是业务处理器去进行处理,则返回true;如果程序员决定不需要再调用其他的组件去处理请求,则返回false。
postHandle():这个方法在业务处理器处理完请求后,但是DispatcherServlet向客户端返回响应前被调用,在该方法中对用户请求request进行处理。
afterCompletion():这个方法在DispatcherServlet 完全处理完请求后被调用,可以在该方法中进行一些资源清理的操作。
public classFirstInterceptor implements HandlerInterceptor{ /** * 该方法在目标方法之前被调用. * 若返回值为 true, 则继续调用后续的拦截器和目标方法. * 若返回值为 false, 则不会再调用后续的拦截器和目标方法. * * 可以考虑做权限. 日志, 事务等. */ @Override public voidafterCompletion(HttpServletRequest arg0, HttpServletResponsearg1, Object arg2, Exception arg3) throws Exception { System.out.println("FirstInterceptor:afterCompletion"); } /** * 调用目标方法之后, 但渲染视图之前. * 可以对请求域中的属性或视图做出修改. */ @Override public voidpostHandle(HttpServletRequest arg0, HttpServletResponse arg1, Objectarg2, ModelAndView arg3) throws Exception { System.out.println("FirstInterceptor:postHandle"); } /** * 在目标方法之前 * 若返回值为true,则继续调用后续拦截器和目标方法 * 若返回值为false,则不会 */ @Override public booleanpreHandle(HttpServletRequest arg0, HttpServletResponse arg1, Objectarg2) throwsException { System.out.println("FirstInterceptor:preHandle"); return true; } }2. 拦截器方法执行顺序
3. 配置自定义拦截器
<mvc:interceptors> <!-- 配置LocaleChanceInterceptor --> <bean class="org.springframework.web.servlet.i18n.LocaleChangeInterceptor"></bean> <!-- 配置自定义拦截器 --> <bean class="spring.interceptors.FirstInterceptor"></bean> <!-- 配置拦截器(不)作用的路径 --> <mvc:interceptor> <mvc:mapping path="/emps"/> <bean class="spring.interceptors.SecondInterceptor"></bean> </mvc:interceptor> </mvc:interceptors>public classSecondInterceptor implements HandlerInterceptor{ @Override public booleanpreHandle(HttpServletRequest request, HttpServletResponseresponse, Object handler) throws Exception { System.out.println("[SecondInterceptor] preHandle"); return false; } @Override public void postHandle(HttpServletRequestrequest, HttpServletResponseresponse, Object handler, ModelAndViewmodelAndView) throwsException { System.out.println("[SecondInterceptor] postHandle"); } @Override public voidafterCompletion(HttpServletRequest request, HttpServletResponseresponse, Object handler, Exception ex) throws Exception { System.out.println("[SecondInterceptor] afterCompletion"); } }4.拦截器顺序
十三、异常处理
1.异常处理
SpringMVC 通过HandlerExceptionResolver 处理程序的异常,包括 Handler 映射、数据绑定以及目标方法执行时发生的异常。
SpringMVC提供的 HandlerExceptionResolver 的实现类
2. HandlerExceptionResolver
DispatcherServlet默认装配的 HandlerExceptionResolver:
没有使用<mvc:annotation-driven/> 配置:
使用了<mvc:annotation-driven/> 配置– :
主要处理 Handler 中用 @ExceptionHandler注解定义的方法。
第一种方式:
@RequestMapping("/testExceptionHandlerExceptionResolver") public StringtestExceptionHandlerExceptionResolver(@RequestParam("i")inti){ System.out.println("result: "+ (10 / i)); return "success"; } /** * 1. 在@ExceptionHandler方法的入参中可以加入 Exception 类型的参数, 该参数即对应发生的异常对象 * 2. @ExceptionHandler方法的入参中不能传入 Map. 若希望把异常信息传导页面上, 需要使用 ModelAndView 作为返回值 * 3. @ExceptionHandler方法标记的异常有优先级的问题. * 4. @ControllerAdvice: 如果在当前 Handler 中找不到@ExceptionHandler方法来出来当前方法出现的异常, * 则将去@ControllerAdvice标记的类中查找@ExceptionHandler标记的方法来处理异常. */ @ExceptionHandler({ArithmeticException.class}) public ModelAndViewhandleArithmeticException(Exception ex){ System.out.println("出异常了: " + ex); ModelAndViewmv = newModelAndView("error"); mv.addObject("exception",ex); return mv; }@ExceptionHandler 注解定义的方法优先级问题:
例如发生的是NullPointerException,但是声明的异常有RuntimeException 和 Exception,此候会根据异常的最近继承关系找到继承深度最浅的那个@ExceptionHandler注解方法,即标记了 RuntimeException 的方法
ExceptionHandlerMethodResolver内部若找不到@ExceptionHandler 注解的话,会找
@ControllerAdvice 中的@ExceptionHandler注解方法
第二种方式(推荐):
@ControllerAdvicepublic classSpringMVCTestExceptionHandler { @ExceptionHandler({ArithmeticException.class}) public ModelAndViewhandleArithmeticException(Exception ex){ System.out.println("----> 出异常了: " + ex); ModelAndViewmv = newModelAndView("error"); mv.addObject("exception",ex); return mv; }}3. ResponseStatusExceptionResolver
在异常及异常父类中找到 @ResponseStatus注解,然后使用这个注解的属性进行处理。定义一个@ResponseStatus 注解修饰的异常类
@ResponseStatus(value=HttpStatus.FORBIDDEN, reason="用户名密码不匹配!")public classUserNameNotMatchPassWordException extends RuntimeException{ private static final long serialVersionUID= 1L; }若在处理器方法中抛出了上述异常:
若ExceptionHandlerExceptionResolver不解析述异常。由于触发的异常UnauthorizedException 带有@ResponseStatus注解。因此会被ResponseStatusExceptionResolver解析到。最后响应HttpStatus.UNAUTHORIZED 代码给客户
端。HttpStatus.UNAUTHORIZED 代表响应码401,无权限。关于其他的响应码请参考 HttpStatus 枚举类型源码。
//@ResponseStatus(reason="测试",value=HttpStatus.NOT_FOUND) @RequestMapping("/testResponseStatusExceptionResolver") public StringtestResponseStatusExceptionResolver(@RequestParam("i")inti){ if(i == 13){ throw newUserNameNotMatchPasswordException(); } System.out.println("testResponseStatusExceptionResolver..."); return "success"; }4. DefaultHandlerExceptionResolver
对一些特殊的异常进行处理,比如NoSuchRequestHandlingMethodException、
HttpRequestMethodNotSupportedException、HttpMediaTypeNotSupportedException、HttpMediaTypeNotAcceptableException等。
@RequestMapping(value="/testDefaultHandlerExceptionResolver",method=RequestMethod.POST) public StringtestDefaultHandlerExceptionResolver(){ System.out.println("testDefaultHandlerExceptionResolver..."); return "success"; }5. 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>@RequestMapping("/testSimpleMappingExceptionResolver") public StringtestSimpleMappingExceptionResolver(@RequestParam("i")inti){ String[] vals = newString[10]; System.out.println(vals[i]); return "success"; }<h4>error page</h4> ${requestScope.ex }
源代码文件:http://download.csdn.net/detail/QQ_26553781/9757893
新闻热点
疑难解答