前言
-
过滤器(Filter):tomcat级别,对Request请求起到过滤的作⽤,作⽤在Servlet之前,如果配置为/*可以对所 有的资源访问(servlet、js/css静态资源等)进⾏过滤处理
-
监听器(Listener):实现了javax.servlet.ServletContextListener 接⼝的服务器端组件,它随 Web应⽤的启动⽽启动,只初始化⼀次,然后会⼀直运⾏监视,随Web应⽤的停⽌⽽销毁
- 作用1:做⼀些初始化⼯作,web应⽤中spring容器启动ContextLoaderListener
- 作用2:监听web中的特定事件,比如HttpSession,ServletRequest的创建和销毁;变量的创建、 销毁和修改等。可以在某些动作前后增加处理,实现监控,比如统计在线⼈数,利用HttpSessionLisener等。
-
拦截器(Interceptor):是SpringMVC自己的,不会拦截 jsp/html/css/image的访问等,只会拦截访问的控制器方法(Handler)。
在请求到来和返回响应的过程中,servlet、filter、listener的执行流程:
- 在Handler业务逻辑执⾏之前拦截⼀次
- 在Handler逻辑执⾏完毕但未跳转⻚⾯之前拦截⼀次
- 在跳转页面之后拦截⼀次
一、单个拦截器的执行流程
- 程序先执行preHandle()方法,如果该方法返回值为true,则程序会继续向下执行处理器中的方法,否则不会向下执行
- handler处理完请求后,执行postHandle()方法,然后会通过DispatcherServlet向客户端返回响应
- 在DispatcherServlet处理完请求后,才会执行afterCompletion()方法。
二、单个拦截器的使用
-
创建springmvc工程,引入相关的依赖,配置web.xml和springmv.xml文件,创建对应的controller层,详情见springmvc的基本应用
-
自定义SpringMVC拦截器,继承HandlerInterceptor
public class MyInterceptor1 implements HandlerInterceptor { @Override //在handler执行前执行 public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("prehandle方法执行了..."); //返回true才会放行 return true; } @Override //可以在modelAndView中做一些视图渲染的处理,在handler执行后视图渲染前执行。 public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("postHandle方法执行了..."); } @Override //在视图渲染后执行,可以捕获处理异常,现在通常使用全局异常捕获处理 public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("afterCompletion方法执行了..."); } }
-
注册SpringMVC拦截器
<mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.lb.interceptor.MyInterceptor1"/> </mvc:interceptor> </mvc:interceptors>
-
测试
三、多个拦截器的执行流程
由图可见,prehandle()方法按照配置顺序执行,其他方法按照反序执行
四、多个拦截器的使用
-
再添加一个自定义Interceptor,修改其中的打印信息
public class MyInterceptor2 implements HandlerInterceptor { @Override //在handler执行前执行 public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("MyInterceptor2的prehandle方法执行了..."); //返回true才会放行 return true; } @Override //可以在modelAndView中做一些视图渲染的处理,在handler执行后视图渲染前执行。 public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("MyInterceptor2的postHandle方法执行了..."); } @Override //在视图渲染后执行,可以捕获处理异常,现在通常使用全局异常捕获处理 public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("MyInterceptor2的afterCompletion方法执行了..."); } }
-
在springmvc.xml中注册第二个拦截器
<mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <!--<mvc:exclude-mapping path="/demo/**"/>--> <bean class="com.lb.interceptor.MyInterceptor1"/> </mvc:interceptor> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.lb.interceptor.MyInterceptor2"/> </mvc:interceptor> </mvc:interceptors>
-
执行测试方法
总结
- 过滤器是tomcat级别的,在web.xml中定义;而拦截器是springmvc级别的,在springmvc.xml中定义。
- 定义Interceptor再注册进springmvc.xml即可使用。
- 需要注意单个和多个拦截器在整个请求过程中的执行顺序。
- springmvc拦截器的经典使用场景:登录拦截。