一般情况下,当 SpringMVC 收到请求时,DispatcherServlet 会将请求交给处理器映射(HandlerMapping),让它找出对应该请求的 HandlerExecutionChain 对象。在此之前我们还可以拦截器对请求消息进行处理,所以在介绍 HandlerMapping 之前,我们有必要认识一下这个 HandlerExecutionChain 对象。
HandlerExecutionChain 顾名思义是一个执行链,它包含一个处理该请求的处理器(Handler),同时可以包括若干个对该请求实施拦截的拦截器(HandlerInterceptor)。当 HandlerMapping 返回 HandlerExecutionChain 后,DispatcherServlet 将请求交给定义在 HandlerExecutionChain 中的拦截器和处理器一并处理。
HandlerExecutionChain 是负责处理请求并返回 ModelAndView 的处理执行链,其结构如图。
请求在被 Handler 调用前后,链中装配的 HandlerInterceptor 会实施拦截操作。
SpringMVC 也可以使用自定义的拦截器对请求进行拦截处理,以达到实现特定功能的作用。
自定义拦截器必须实现 HandlerInterceptor 接口,或者继承实现了 HandlerInterceptor 接口的抽象类 HandlerInterceptorAdapter。
它包含方法有:
1. boolean preHandle(HttpServletRequest request, HttpServletReponse reponse, Object handler):这个方法在业务处理器处理请求之前被调用,在该方法中对用户请求 request 进行处理。如果程序员决定该拦截器对请求进行拦截处理后还要调用其它的拦截器,或者是业务处理器去进行处理,则返回 true;如果程序员决定不需要再调用其它的组件去处理请求,则返回 false。可以考虑做权限、日志、事务等。
2. void postHandle(HttpServletRequest request, HttpServletReponse reponse, Object handler, ModelAndView modelAndView):这个方法在业务处理器处理完请求后,但 DispatcherServlet 向客户端返回响应前被调用。在该方法中对用户请求 request 进行处理。可以对请求域中的属性或试图做出修改。
3. void afterCompletion(HttpServletRequest request, HttpServletReponse reponse, Object handler, Exception ex):这个方法在 DispatcherServlet 完全处理完请求后被调用,可以在该方法中进行一些资源清理的操作。
编写自定义的拦截器如下代码所示
package com.bupt.springmvc.converter.interceptor; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; public class FirstInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("FirstInterceptor PreHandler..."); return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("FirstInceptor PostHandler..."); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("FirstInterceptor AfterCompletion..."); } }
同时需要在springmvc配置文件中声明这个拦截器
<mvc:interceptors>
<bean class="com.bupt.springmvc.converter.interceptor.FirstInterceptor"/>
</mvc:interceptors>
当正常访问某个请求时,拦截器会产生作用,控制台输出为
此时拦截器执行的顺序可以表示为
在SpingMVC配置文件中配置拦截器时,还可以通过 <mvc:intercpetor> 属性来指定拦截器只能或者不能作用于哪些请求路径。
当配置多个拦截器时,在配置文件中注册的前后顺序就是其执行时的先后顺序。
如:当配置如下,FirstInterceptor 注册顺序早于 SecondInterceptor,执行时则先执行 FirstInterceptor 再执行 SecondInterceptor。
<mvc:interceptors> <bean class="com.bupt.springmvc.converter.interceptor.FirstInterceptor"/> <mvc:interceptor> <!-- 指定SecondInterceptor拦截器作用于/emps请求路径不作用于/emp请求路径 --> <mvc:mapping path="/emps"/> <mvc:exclude-mapping path="/emp"/> <bean class="com.bupt.springmvc.converter.interceptor.SecondInterceptor"/> </mvc:interceptor> </mvc:interceptors>
当某个请求配置了两个拦截器,如上面的配置,当我访问 /emps 请求路径时,控制台会得出如下的结果
整个过程可以用如下流程图表示
以上是两个拦截器的 preHandler 方法返回 ture 时,程序的执行结果。
当 FirstInterceptor 的 preHandler 方法返回 false ,控制台输出如图,它只执行了 preHandler 方法。并且页面没有跳转,没有相应的处理方法得到执行
当 SecondInterceptor 的 preHandler 方法返回 false 时,程序执行的结果和流程如下所示
之前的一些方法都不会执行了,只执行如下的方法