拦截器VS过滤器:Spring Boot中请求处理的艺术!

devtools/2025/2/21 8:55:09/

在这里插入图片描述

目录

    • 一、拦截器(Interceptor)和过滤器(Filter):都是“守门员”!
    • 二、如何实现拦截器和过滤器?
    • 三、拦截器和过滤器的区别
    • 四、执行顺序
    • 五、真实的应用场景
    • 六、总结

🌟如果喜欢作者的讲解方式,关注作者不迷路,同时也可以看看我的其他文章! 感谢!!!
🌟 从乐高积木到乐队指挥,用最通俗易懂的方式带你玩转 Spring Boot Bean!

就让我用这篇文章来讲解 SpringBoot 的拦截器和过滤器吧,给它安排明白!😎

一、拦截器(Interceptor)和过滤器(Filter):都是“守门员”!

想象一下,你的 SpringBoot 应用就像一个豪华酒店🏨。

  • 过滤器(Filter): 就像酒店大门口的保安👮,负责检查所有进出酒店的人(请求)。它可以决定是否允许客人进入,或者在客人进入前做一些处理,比如检查身份证、测量体温🌡️。
  • 拦截器(Interceptor): 就像酒店内部各个楼层的楼层经理👩‍💼,只负责检查进入特定楼层(Controller)的客人。它可以在客人进入楼层前、进入楼层后、离开楼层后都进行干预,比如登记访客信息、提供楼层指引🗺️。

二、如何实现拦截器和过滤器?

1. 过滤器(Filter):

  • 步骤一: 创建一个类,实现 javax.servlet.Filter 接口。
  • 步骤二: 实现 doFilter() 方法。这个方法就是过滤器的核心逻辑,在这里你可以对请求和响应进行处理。
  • 步骤三: 使用 @WebFilter 注解或者在 WebConfig 类中注册过滤器。
java">// 方式一:使用 @WebFilter 注解
@WebFilter(urlPatterns = "/*", filterName = "MyFilter") // 拦截所有请求
public class MyFilter implements Filter {@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {System.out.println("过滤器:请求来了!我要检查一下!🧐");// 可以对 request 和 response 进行处理HttpServletRequest req = (HttpServletRequest) request;String uri = req.getRequestURI();System.out.println("请求的URI:" + uri);// 放行,让请求继续往下走chain.doFilter(request, response);System.out.println("过滤器:请求走了!我要记录一下!📝");}
}// 方式二:在 WebConfig 类中注册
@Configuration
public class WebConfig {@Beanpublic FilterRegistrationBean<MyFilter> myFilterRegistrationBean() {FilterRegistrationBean<MyFilter> registration = new FilterRegistrationBean<>();registration.setFilter(new MyFilter());registration.addUrlPatterns("/*"); // 拦截所有请求registration.setName("MyFilter");registration.setOrder(1); // 设置优先级,数字越小优先级越高return registration;}
}

2. 拦截器(Interceptor):

  • 步骤一: 创建一个类,实现 org.springframework.web.servlet.HandlerInterceptor 接口。
  • 步骤二: 实现 preHandle()postHandle()afterCompletion() 方法。
    • preHandle():在 Controller 处理请求之前调用,可以进行权限验证、参数校验等。如果返回 false,则请求会被拦截,不会继续执行。
    • postHandle():在 Controller 处理请求之后,但在视图渲染之前调用,可以对 ModelAndView 进行修改。
    • afterCompletion():在整个请求处理完毕之后调用,可以进行资源清理、日志记录等。
  • 步骤三: 创建一个配置类,实现 org.springframework.web.servlet.config.annotation.WebMvcConfigurer 接口,并重写 addInterceptors() 方法,将拦截器注册到 Spring 容器中。
java">public class MyInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("拦截器:Controller 要处理请求了!我要先检查一下!👮‍♀️");// 可以进行权限验证、参数校验等String token = request.getHeader("token");if (token == null || token.isEmpty()) {System.out.println("拦截器:没有 token,拒绝访问!🚫");response.setStatus(401); // 返回未授权状态码return false; // 拦截请求}return true; // 放行请求}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("拦截器:Controller 处理完请求了!我可以修改 ModelAndView!🎨");// 可以对 ModelAndView 进行修改if (modelAndView != null) {modelAndView.addObject("message", "拦截器添加的额外信息!🎁");}}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("拦截器:请求处理完成了!我可以清理资源、记录日志!🧹");// 可以进行资源清理、日志记录等}
}@Configuration
public class InterceptorConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new MyInterceptor()).addPathPatterns("/api/**") // 拦截 /api/ 下的所有请求.excludePathPatterns("/api/login"); // 排除 /api/login 请求}
}

三、拦截器和过滤器的区别

特性过滤器(Filter)拦截器(Interceptor)
实现方式实现 javax.servlet.Filter 接口实现 org.springframework.web.servlet.HandlerInterceptor 接口
拦截范围拦截所有进出 Servlet 容器的请求拦截特定的 Controller 方法
执行时机在 DispatcherServlet 之前和之后执行在 Controller 方法执行之前、之后和完成之后执行
依赖性不依赖 Spring 容器依赖 Spring 容器,可以访问 Spring 上下文
功能字符编码转换、请求内容过滤、敏感词过滤等权限验证、日志记录、参数校验等
精确度粗粒度,只能拦截 URL细粒度,可以访问 HandlerMethod,获取方法信息

总结:

  • 过滤器: 拦截所有请求,不依赖 Spring,功能比较通用。
  • 拦截器: 拦截特定 Controller 方法,依赖 Spring,功能更精细。

四、执行顺序

  1. 过滤器(Filter):在 DispatcherServlet 之前执行。
  2. 拦截器(Interceptor)
    • preHandle():在 Controller 方法执行之前执行。
    • Controller 方法执行。
    • postHandle():在 Controller 方法执行之后,但在视图渲染之前执行。
    • 视图渲染。
    • afterCompletion():在整个请求处理完毕之后执行。

记住: 多个过滤器和拦截器可以配置执行顺序,通常通过 order 属性或者 @Order 注解来设置,数字越小优先级越高。

五、真实的应用场景

  • 过滤器:
    • 字符编码转换: 统一设置请求和响应的字符编码,避免乱码。
    • XSS 攻击防御: 过滤请求中的恶意脚本,防止 XSS 攻击。
    • 日志记录: 记录所有请求的 URL、IP 地址等信息。
  • 拦截器:
    • 权限验证: 检查用户是否已登录,是否有权限访问特定资源。
    • 参数校验: 校验请求参数的合法性,防止恶意请求。
    • 性能监控: 记录 Controller 方法的执行时间,分析性能瓶颈。
    • 防止重复提交: 避免用户重复提交表单。

六、总结

  • 过滤器: 就像酒店大门口的保安,啥人都拦,主要做一些通用的检查,比如身份证、体温啥的。
  • 拦截器: 就像楼层经理,只管特定楼层的人,可以做更细致的检查,比如登记访客信息、提供楼层指引。
  • 执行顺序: 先保安,再楼层经理。保安先检查,楼层经理在客人进房间前、进房间后、离开房间后都可以管。

如果编写了AOP逻辑,会在拦截器之后执行,相关内容请看:
🔗被重复代码逼疯?AOP来当“舔狗”!日志/事务/权限,随叫随到!

希望这篇文章能让你有趣的理解 SpringBoot 的拦截器和过滤器!🎉🎉🎉


http://www.ppmy.cn/devtools/160646.html

相关文章

Chrome Edge 开启多线程下载

开启步骤 Chrome浏览器 地址栏输入并回车&#xff1a;chrome://flags/#enable-parallel-downloading Edge浏览器(Chromium 内核) 地址栏输入并回车&#xff1a;edge://flags/#enable-parallel-downloading 将默认的 Default 改成 Enabled 然后重启即可

蓝桥杯备赛1-5大写

题目描述 给定一个只包含大写字母和小写字母的字符串&#xff0c;请将其中所有的小写字母转换成大写字母后将字符串输出。 输入描述 输入一行包含一个字符串。 输出描述 输出转换成大写后的字符串。 输入输出样例 示例 输入 LanQiao输出 LANQIAO评测用例规模与约定 对于…

如何写出优秀的测试用例?

一、测试点与测试用例 测试点不等于测试用例&#xff0c;这是我们首先需要认识到的。 问题1&#xff1a;这些测试点在内容上有重复&#xff0c;存在冗余。 问题2&#xff1a;一些测试点的测试输入不明确&#xff0c;不知道测试时要测试哪些。 问题3&#xff1a;总是在搭相似…

基于SpringBoot的驾校报名小程序系统设计与实现(源码+文档)

大家好&#xff0c;今天要和大家聊的是一款基于SpringBoot的“驾校报名小程序”系统的设计与实现。项目源码以及部署相关事宜请联系我&#xff0c;文末附上联系方式。 项目简介 基于SpringBoot的“驾校报名小程序”系统设计与实现的主要使用者分为管理员、驾校教练和用户&…

服务保护和分布式事务

背景 雪崩问题 原因 最开始只有商品服务有问题&#xff0c;后来购物车服务和其他服务都陷入了瘫痪状态 原因总结 解决方案 服务保护方案-请求限流&#xff08;被调用者&#xff09; 请求限流&#xff1a;限制访问微服务的请求的并发量&#xff0c;避免服务因流量激增出现故障…

ECMAScript6------数组扩展

ECMAScript6------数组扩展 1.扩展运算符1.1 含义1.2 应用(1)复制数组(浅拷贝)(2)合并数组(3)与解构赋值结合(4)字符串(5)实现了Iterator接口的对象(6)Map和Set结构,Generator 函数 2.Array.from和Array.of(1)Array.from(2)Array.of 3.新增的实例方法(1)copyWithin()(2)find(回…

linux学习【7】Sourc Insight 4.0设置+操作

目录 1.Source Insight是什么&#xff1f;2.需要哪些配置&#xff1f;3.怎么新建项目4.一些问题的解决1.中文乱码问题 按照这个设置就可以了&#xff0c;下面的设置会标明设置理由。 1.Source Insight是什么&#xff1f; 阅读源码&#xff0c;编辑源码&#xff0c;不能编译&am…

SOME/IP--协议英文原文讲解12(完结)

前言 SOME/IP协议越来越多的用于汽车电子行业中&#xff0c;关于协议详细完全的中文资料却没有&#xff0c;所以我将结合工作经验并对照英文原版协议做一系列的文章。基本分三大块&#xff1a; 1. SOME/IP协议讲解 2. SOME/IP-SD协议讲解 3. python/C举例调试讲解 4.3 Compa…