拦截器的原理很简单,是 AOP 的一种实现,专门拦截对动态资源的后台请求,即拦截对控制层的请求。常见的使用场景包括判断用户是否有权限请求后台,再拔高一层的使用场景,比如拦截器可以结合 WebSocket 一起使用,用来拦截 WebSocket 请求,并做相应的处理等等。

拦截器不会拦截静态资源,Spring Boot 的默认静态目录为 resources/static,该目录下的静态页面、JS、CSS、图片等等,不会被拦截,当然这也要看实现过程,有些情况下可能会被拦截,在下文会指出。

定义拦截器

定义拦截器,只需要实现 HandlerInterceptor 接口。HandlerInterceptor 接口是所有自定义拦截器或者 Spring Boot 提供的拦截器的鼻祖,所以,首先来了解下该接口。该接口中有三个方法,分别为 :

preHandle(……)、postHandle(……) 和 afterCompletion(……)-----------------------------------------------------------------------------------------preHandle(……) 方法:
该方法的执行时机是,当某个 URL 已经匹配到对应的 Controller 中的某个方法,且在这个方法执行之前。
所以 preHandle(……) 方法可以决定是否将请求放行,这是通过返回值来决定的,返回 true 则放行,返回 false 则不会向后执行。
-----------------------------------------------------------------------------------------
postHandle(……) 方法:
该方法的执行时机是,当某个 URL 已经匹配到对应的 Controller 中的某个方法,且在执行完了该方法,
但是在 DispatcherServlet 视图渲染之前。所以在这个方法中有个 ModelAndView 参数,可以在此做一些修改动作。
-----------------------------------------------------------------------------------------
afterCompletion(……) 方法:
顾名思义,该方法是在整个请求处理完成后(包括视图渲染)执行,这时做一些资源的清理工作,
这个方法只有在 preHandle(……) 被成功执行后并且返回 true 才会被执行。

接下来自定义一个拦截器。代码如下:

/*** 自定义拦截器* @author shengwu ni* @date 2018/08/03*/
public class MyInterceptor implements HandlerInterceptor {private static final Logger logger = LoggerFactory.getLogger(MyInterceptor.class);@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {HandlerMethod handlerMethod = (HandlerMethod) handler;Method method = handlerMethod.getMethod();String methodName = method.getName();logger.info("====拦截到了方法:{},在该方法执行之前执行====", methodName);// 返回true才会继续执行,返回false则取消当前请求return true;}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {logger.info("执行完方法之后进执行(Controller方法调用之后),但是此时还没进行视图渲染");}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {logger.info("整个请求都处理完咯,DispatcherServlet也渲染了对应的视图咯,此时我可以做一些清理的工作了");}
}

拦截器已经定义完成,接下来就是对该拦截器进行拦截配置。

配置拦截器

在 Spring Boot 2.0 之前,我们都是直接继承 WebMvcConfigurerAdapter 类,然后重写 addInterceptors 方法来实现拦截器的配置。但是在 Spring Boot 2.0 之后,该方法已经被废弃了(当然,也可以继续用),取而代之的是 WebMvcConfigurationSupport 方法,如下:

@Configuration
public class MyInterceptorConfig extends WebMvcConfigurationSupport {@Overrideprotected void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**");super.addInterceptors(registry);}
}

在该配置中重写 addInterceptors 方法,将我们上面自定义的拦截器添加进去,addPathPatterns 方法用来添加要拦截的请求,这里我们拦截所有的请求。现在我们配置好了拦截器,接下来写一个 Controller 测试一下:

@Controller
@RequestMapping("/interceptor")
public class InterceptorController {@RequestMapping("/test")public String test() {return "hello";}
}

解决静态资源被拦截问题

也就是说,虽然 Spring Boot 2.0 废弃了 WebMvcConfigurerAdapter,但是 WebMvcConfigurationSupport 又会导致默认的静态资源被拦截,这就需要我们手动将静态资源放开。

如何放开呢?除了在 MyInterceptorConfig 配置类中重写 addInterceptors 方法,还需要再重写一个方法 addResourceHandlers,用来将静态资源放开:

/*** 用来指定静态资源不被拦截,否则继承WebMvcConfigurationSupport这种方式会导致静态资源无法直接访问* @param registry*/
@Override
protected void addResourceHandlers(ResourceHandlerRegistry registry) {registry.addResourceHandler("/**").addResourceLocations("classpath:/static/");super.addResourceHandlers(registry);
}

如上配置好之后,重启项目,静态资源也可以正常访问了。如果你是个善于学习或者研究的人,那肯定不会止步于此,没错,上面这种方式的确能解决静态资源无法访问的问题,但是,还有更方便的配置方式。

我们不继承 WebMvcConfigurationSupport 类,直接实现 WebMvcConfigurer 接口,然后重写 addInterceptors 方法,将自定义的拦截器添加进去即可,如下:

@Configuration
public class MyInterceptorConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {// 实现WebMvcConfigurer不会导致静态资源被拦截registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**");}
}

这样就非常方便了,通过实现 WebMvcConfigure 接口,使 Spring Boot 默认的静态资源不会拦截。

这两种方式都可以,两者更多具体细节,感兴趣的读者可以进一步研究。由于这两种方式的不同,继承 WebMvcConfigurationSupport 类的方式可以用在前后端分离的项目中,后台不需要访问静态资源(就不需要放开静态资源了);实现 WebMvcConfigure 接口的方式可以用在非前后端分离的项目中,因为需要读取一些图片、CSS、JS 文件等等。

拦截器使用实例

判断用户有没有登录

一般用户的登录功能,我们可以这么实现,要么在 Session 中写一个 user,要么针对每个 user 生成一个 Token,相比之下,第二种要更好。

第二种方式中,如果用户登录成功,每次请求时都会带上该用户的 Token,如果未登录,则没有该 Token,服务端可以检测这个 Token 参数的有无来判断用户有没有登录,从而实现拦截功能。我们改造一下 preHandle 方法,如下:

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {HandlerMethod handlerMethod = (HandlerMethod) handler;Method method = handlerMethod.getMethod();String methodName = method.getName();logger.info("====拦截到了方法:{},在该方法执行之前执行====", methodName);// 判断用户有没有登陆,一般登陆之后的用户都有一个对应的tokenString token = request.getParameter("token");if (null == token || "".equals(token)) {logger.info("用户未登录,没有权限执行……请登录");return false;}// 返回true才会继续执行,返回false则取消当前请求return true;
}

重启项目,在浏览器中输入:localhost:8080/interceptor/test, 之后查看控制台日志,发现被拦截,如果在浏览器中输入: localhost:8080/interceptor/test?token=123 即可正常往下访问。

取消拦截操作

根据上文,如果我要拦截所有 /admin 开头的 URL 请求,需要在拦截器配置中添加该前缀。但在实际项目中,可能会有这种场景出现:某个请求也是 /admin 开头,但不能拦截,比如 /admin/login 等等,这样的话还需再去配置。这时,可不可以做成一个类似于开关的东西,哪里不需要拦截,我就在哪里放个开关,做成灵活的可插拔的效果呢?

可以的,我们可以定义一个注解,该注解专门用来取消拦截操作,如果某个 Controller 中的方法我们不需要拦截,即可在该方法上加上我们自定义的注解即可,下面先定义一个注解:

/*** 该注解用来指定某个方法不用拦截*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface UnInterception {
}

然后在 Controller 中的某个方法上添加该注解,在拦截器处理方法中添加该注解取消拦截的逻辑,如下:

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {HandlerMethod handlerMethod = (HandlerMethod) handler;Method method = handlerMethod.getMethod();String methodName = method.getName();logger.info("====拦截到了方法:{},在该方法执行之前执行====", methodName);// 通过方法,可以获取该方法上的自定义注解,然后通过注解来判断该方法是否要被拦截// @UnInterception 是我们自定义的注解UnInterception unInterception = method.getAnnotation(UnInterception.class);if (null != unInterception) {return true;}// 返回true才会继续执行,返回false则取消当前请求return true;
}

Controller 中的方法代码可以参见源码,重启项目在浏览器中输入: http://localhost:8080/interceptor/test2?token=123, 测试一下,可以看出,加了该注解的方法不会被拦截。

总结:

Spring Boot 2.0 之后版本中拦截器的配置支持两种方式,可以根据实际情况选择不同的配置方式。

Spring Boot细节挖掘(拦截器)相关推荐

  1. Spring Boot 实现登录拦截器,这才是正确的姿势!!

    原文:https://blog.csdn.net/qq_27198345/article/details/111401610 对于管理系统或其他需要用户登录的系统,登录验证都是必不可少的环节,在Spr ...

  2. Spring Boot实战:拦截器与过滤器

    一.拦截器与过滤器 在讲Spring boot之前,我们先了解一下过滤器和拦截器.这两者在功能方面很类似,但是在具体技术实现方面,差距还是比较大的.在分析两者的区别之前,我们先理解一下AOP的概念,A ...

  3. Spring Boot细节挖掘(Redis的集成)

    Spring Boot 对 Redis 的支持已经非常完善,丰富的 API 足够我们日常的开发,这里我介绍几个最常用的供大家学习,其他 API 希望大家自己多学习,多研究,用到会去查即可. 目前有两个 ...

  4. Spring Boot 过滤器与拦截器的使用及其区别 和过滤器怎么设置运行顺序

    过滤器 过滤器的定义 在客户端将请求发送到服务器端之前,拦截这些请求:在服务器端的响应返回给客户端之前,处理这些响应. 比如字符编码过滤器CharacterEncodingFilter,就是设置req ...

  5. Spring Boot笔记-设置拦截器为false时返回的Body

    以preHandle拦截器为例: @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse ...

  6. Spring Boot细节挖掘(Docker部署项目)

    如今 Docker 的使用已经非常普遍,特别是在一线互联网公司,使用 Docker 技术可以帮助企业快速水平扩展服务,从而达到弹性部署业务的能力.在云服务概念兴起之后,Docker 的使用场景和范围进 ...

  7. Spring Boot 细节挖掘(全局异常处理)

    对于全局异常处理app开发很重要 直接上步骤: 第一创建JsonResult: package com.swaager.swaagerdemo.model;/*** @authorseerhuitao ...

  8. spring过滤器Filter 、 拦截器Interceptor 、 切片Aspect 详解

    springboot 过滤器Filter vs 拦截器Interceptor vs 切片Aspect 详解 1 前言 最近接触到了过滤器和拦截器,网上查了查资料,这里记录一下,这篇文章就来仔细剖析下过 ...

  9. spring 两次进入拦截器_过滤器和拦截器的 6 个区别,别再傻傻分不清了

    点击上方 肉眼品世界,选择 设为星标 深度价值体系传递 作者 :程序员内点事 来源 :toutiao.com/i6834310440495874563 毕竟这两种工具开发中用到的频率都相当高,应用起来 ...

最新文章

  1. 将 Shiro 作为一个许可为基础的应用程序 五:password加密/解密Spring应用
  2. 成功解决在excel表中通过数学函数转换后,接着去掉公式转为不再随着变化的数值
  3. DayDayUp:大学英语六级考试简介、应试分析、题型内容((Writing/Listening/Reading/Translation)、解答经验步骤、解题技巧之最强攻略(建议收藏)
  4. gpg加密命令 linux_用 PGP 保护代码完整性(四):将主密钥移到离线存储中 | Linux 中国...
  5. spring el 表达式的上下文关联到 ApplicationContext
  6. 关于ABAP排序不稳定的问题
  7. qt中创建控件布局以及删除原有布局和控件
  8. STM32之SPI原理
  9. 获取HG526超级密码
  10. Modbus RTU CRC校验码计算方法
  11. java连接mysql,报错Could not create connection to database server.
  12. 金山云服务器e1型,金山云-文档中心-重装系统
  13. kmeans设置中心_kmeans算法
  14. 计算机函数求各科及格率怎么求,合格率怎么算(计算及格率的方法)
  15. 对接微信支付退款证书之坑
  16. 如何选取最佳前缀索引长度
  17. Eclipse中调试Python代码--调试FWTools2.4.7中的gdal_retile.py
  18. uts大学计算机排名,uts是什么大学
  19. 机器学习 day5 day6 分类问题实战:判断是否为羊毛党
  20. Java毕设 仿京东淘宝 多用户商城平台 毕业设计源码 使用教程(2)店铺功能

热门文章

  1. Android-入门学习笔记-JSON 解析
  2. 构建微服务(Building Microservices)-PDF 文档
  3. Android应用程序文件缓存getCacheDir()和getExternalCacheDir()
  4. Android使用软引用和弱引用
  5. 封装好的实用的读写XML类---增删改查XML
  6. 第一个JSP文件的创建过程
  7. linux 启动tomcat 怎么显示日志文件,随着LINUX的启动,打开一个终端显示TOMCAT的日志文件,请问如何做到?...
  8. java esclient query_elasticsearch 口水篇(4)java客户端 - 原生esClient
  9. 怎么用python算单价和总价_用python计算最高投标限价
  10. 用servlet路径访问一个html,java web中servlet、jsp、html 互相访问的路径问题