拦截器

1.简介

Spring MVC 中的拦截器(Interceptor)类似于 Servlet 开发中的过滤器 Filter,它主要用于拦截用户请求并作相应的处理,它也是 AOP 编程思想的体现,底层通过动态代理模式完成。

2.定义实现类

拦截器有两种实现方式: 1.实现 HandlerInterceptor 接口 2.继承 HandlerInterceptorAdapter 抽象类(看源码最底层也是通过 HandlerInterceptor 接口 实现)

3.HandlerInterceptor方法介绍

  1. @Override
  2. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  3. throws Exception {
  4. //进行逻辑判断,如果ok就返回true,不行就返回false,返回false就不会处理请求
  5. return true;
  6. }
  7. @Override
  8. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  9. ModelAndView modelAndView) throws Exception {
  10. }
  11. @Override
  12. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
  13. throws Exception {
  14. }
  15. 复制代码

preHandle:在业务处理器处理请求之前被调用。预处理,可以进行编码、安全控制、权限校验等处理; postHandle:在业务处理器处理请求执行完成后,生成视图之前执行。 afterCompletion:在 DispatcherServlet 完全处理完请求后被调用,可用于清理资源等。

4.应用场景

1.日志记录:记录请求信息的日志,以便进行信息监控、信息统计、计算PV(Page View)等; 2.登录鉴权:如登录检测,进入处理器检测检测是否登录; 3.性能监控:检测方法的执行时间; 4.其他通用行为。

5.与 Filter 过滤器的区别

1.拦截器是基于java的反射机制的,而过滤器是基于函数回调。 2.拦截器不依赖于servlet容器,而过滤器依赖于servlet容器。 3.拦截器只能对Controller请求起作用,而过滤器则可以对几乎所有的请求起作用。 4.拦截器可以访问action上下文、值栈里的对象,而过滤器不能访问。 5.在Controller的生命周期中,拦截器可以多次被调用,而过滤器只能在容器初始化时被调用一次。 6.拦截器可以获取IOC容器中的各个bean,而过滤器不行。这点很重要,在拦截器里注入一个service,可以调用业务逻辑。

具体实现

单个拦截器

1.新建拦截器

  1. public class Test1Interceptor implements HandlerInterceptor{
  2. @Override
  3. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  4. throws Exception {
  5. System.out.println("执行preHandle方法-->01");
  6. return true;
  7. }
  8. @Override
  9. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  10. ModelAndView modelAndView) throws Exception {
  11. System.out.println("执行postHandle方法-->02");
  12. }
  13. @Override
  14. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
  15. throws Exception {
  16. System.out.println("执行afterCompletion方法-->03");
  17. }
  18. }
  19. 复制代码

2.配置拦截器

  1. @Configuration
  2. public class WebMvcConfig extends WebMvcConfigurationSupport {
  3. /*
  4. * 拦截器配置
  5. */
  6. @Override
  7. public void addInterceptors(InterceptorRegistry registry) {
  8. // 注册自定义拦截器,添加拦截路径和排除拦截路径
  9. registry.addInterceptor(new Test1Interceptor()) // 添加拦截器
  10. .addPathPatterns("/**") // 添加拦截路径
  11. .excludePathPatterns(// 添加排除拦截路径
  12. "/hello").order(0);//执行顺序
  13. super.addInterceptors(registry);
  14. }
  15. }
  16. 复制代码

3.测试拦截器

  1. @RestController
  2. public class TestController {
  3. @RequestMapping("/hello")
  4. public String getHello() {
  5. System.out.println("这里是Hello");
  6. return "hello world";
  7. }
  8. @RequestMapping("/test1")
  9. public String getTest1() {
  10. System.out.println("这里是Test1");
  11. return "test1 content";
  12. }
  13. @RequestMapping("/test2")
  14. public String getTest2() {
  15. System.out.println("这里是Test2");
  16. return "test2 content";
  17. }
  18. }
  19. 复制代码

4.单个拦截器的执行流程

通过浏览器测试: http://127.0.0.1:8080/hello 结果:

  1. 这里是Hello
  2. 复制代码

http://127.0.0.1:8080/test1http://127.0.0.1:8080/test2 结果:

  1. 执行preHandle方法-->01
  2. 这里是Test1
  3. 执行postHandle方法-->02
  4. 执行afterCompletion方法-->03
  5. 复制代码

多个拦截器

1.新建两个拦截器

Test1Interceptor

  1. public class Test1Interceptor implements HandlerInterceptor{
  2. @Override
  3. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  4. throws Exception {
  5. System.out.println("执行Test1Interceptor preHandle方法-->01");
  6. return true;
  7. }
  8. @Override
  9. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  10. ModelAndView modelAndView) throws Exception {
  11. System.out.println("执行Test1Interceptor postHandle方法-->02");
  12. }
  13. @Override
  14. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
  15. throws Exception {
  16. System.out.println("执行Test1Interceptor afterCompletion方法-->03");
  17. }
  18. }
  19. 复制代码

Test2Interceptor

  1. public class Test2Interceptor extends HandlerInterceptorAdapter{
  2. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  3. throws Exception {
  4. System.out.println("执行Test2Interceptor preHandle方法-->01");
  5. return true;
  6. }
  7. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
  8. ModelAndView modelAndView) throws Exception {
  9. System.out.println("执行Test2Interceptor postHandle方法-->02");
  10. }
  11. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
  12. throws Exception {
  13. System.out.println("执行Test2Interceptor afterCompletion方法-->03");
  14. }
  15. }
  16. 复制代码

2.配置拦截器

  1. @Configuration
  2. public class WebMvcConfig extends WebMvcConfigurationSupport {
  3. /*
  4. * 拦截器配置
  5. */
  6. @Override
  7. public void addInterceptors(InterceptorRegistry registry) {
  8. // 注册自定义拦截器,添加拦截路径和排除拦截路径
  9. registry.addInterceptor(new Test1Interceptor()) // 添加拦截器1
  10. .addPathPatterns("/**") // 添加拦截路径
  11. .excludePathPatterns(// 添加排除拦截路径
  12. "/hello")
  13. .order(0);
  14. registry.addInterceptor(new Test2Interceptor()) // 添加拦截器2
  15. .addPathPatterns("/**") // 添加拦截路径
  16. .excludePathPatterns(// 添加排除拦截路径
  17. "/test1")
  18. .order(1);
  19. super.addInterceptors(registry);
  20. }
  21. }
  22. 复制代码

3.测试拦截器

  1. @RestController
  2. public class TestController {
  3. @RequestMapping("/hello")
  4. public String getHello() {
  5. System.out.println("这里是Hello");
  6. return "hello world";
  7. }
  8. @RequestMapping("/test1")
  9. public String getTest1() {
  10. System.out.println("这里是Test1");
  11. return "test1 content";
  12. }
  13. @RequestMapping("/test2")
  14. public String getTest2() {
  15. System.out.println("这里是Test2");
  16. return "test2 content";
  17. }
  18. }
  19. 复制代码

4.多个拦截器的执行流程

通过浏览器测试: http://127.0.0.1:8080/test2 结果:

  1. 执行Test1Interceptor preHandle方法-->01
  2. 执行Test2Interceptor preHandle方法-->01
  3. 这里是Test2
  4. 执行Test2Interceptor postHandle方法-->02
  5. 执行Test1Interceptor postHandle方法-->02
  6. 执行Test2Interceptor afterCompletion方法-->03
  7. 执行Test1Interceptor afterCompletion方法-->03
  8. 复制代码

通过示例,简单的说多个拦截器执行流程就是先进后出

简单的 token 判断示例

1.拦截器

  1. @Override
  2. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
  3. throws Exception {
  4. System.out.println("执行Test1Interceptor preHandle方法-->01");
  5. String token = request.getParameter("token");
  6. if (StringUtils.isEmpty(token)) {
  7. response.setContentType("text/html");
  8. response.setCharacterEncoding("UTF-8");
  9. response.getWriter().println("token不存在");
  10. return false;
  11. }
  12. return true;
  13. }
  14. 复制代码

2.测试及结果

未传token:

  1. 执行Test1Interceptor preHandle方法-->01
  2. 复制代码

传token:

  1. 执行Test1Interceptor preHandle方法-->01
  2. 页码:1
  3. 页码大小:10
  4. 执行Test1Interceptor postHandle方法-->02
  5. 执行Test1Interceptor afterCompletion方法-->03