Java Web 拦截机制实战指南:Filter 与 Interceptor 深度解析

一、理解核心概念

在 Java Web 开发中,过滤器(Filter)拦截器(Interceptor)是两种核心的请求处理机制。它们虽然都能对请求进行拦截和处理,但定位截然不同:

  • Filter 是 Servlet 容器的"守门人",位于应用最外层
  • Interceptor 是 Spring MVC 的"执法官",位于框架内部

二、Filter:Servlet 容器的第一道防线

2.1 本质与特点

Filter 是 Java Servlet 规范 定义的组件,由 Servlet 容器(如 Tomcat)直接管理,不依赖任何框架,因此具有最强的通用性。

生命周期方法:

方法

触发时机

用途

init()

应用启动时

初始化配置

doFilter()

每次请求

核心处理逻辑

destroy()

应用关闭时

资源释放

执行流程:

请求 → Filter1 → Filter2 → Servlet/Controller → Filter2 → Filter1 → 响应

2.2 Spring Boot 中的实现方式

方式一:注解方式(适合简单场景)

@WebFilter(urlPatterns = "/api/*", filterName = "authFilter") public class AuthFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // 前置处理 System.out.println("认证检查开始"); // 放行到下一个 Filter 或目标资源 chain.doFilter(request, response); // 后置处理(响应返回时执行) System.out.println("认证检查结束"); } } // 在主类开启扫描 @SpringBootApplication @ServletComponentScan public class Application { }

方式二:配置类方式(推荐,更灵活)

@Configuration public class FilterConfig { @Bean public FilterRegistrationBean<AuthFilter> authFilter() { FilterRegistrationBean<AuthFilter> bean = new FilterRegistrationBean<>(); bean.setFilter(new AuthFilter()); bean.addUrlPatterns("/api/*"); bean.setOrder(1); // 控制执行顺序,数字越小越先执行 return bean; } }

2.3 Filter 的典型应用场景

场景

说明

统一编码

设置 UTF-8 编码,解决中文乱码

安全防护

XSS 过滤、SQL 注入检测

跨域处理

添加 CORS 响应头

请求日志

记录所有请求的 IP、路径、耗时

数据压缩

Gzip 压缩响应内容


三、Interceptor:Spring MVC 的精准拦截

3.1 本质与特点

Interceptor 是 Spring MVC 框架 提供的机制,只能拦截被 DispatcherServlet 处理的请求(即映射到 Controller 的请求)。它可以获取 Spring 上下文,与业务逻辑深度集成。

三个核心拦截点:

方法

执行时机

能否终止请求

典型用途

preHandle()

Controller 方法执行前

可返回 false

终止

登录验证、权限校验

postHandle()

Controller 执行后,视图渲染前

无法终止

修改 Model、添加公共数据

afterCompletion()

请求处理完成(含异常)

无法终止

资源清理、异常日志

3.2 完整实现示例

@Component public class PermissionInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 获取请求信息 String uri = request.getRequestURI(); System.out.println("【拦截】请求路径: " + uri); // 权限校验逻辑 HttpSession session = request.getSession(); if (session.getAttribute("user") == null && uri.startsWith("/admin")) { response.sendRedirect("/login"); return false; // 拦截,不继续执行 } return true; // 放行 } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { // 向视图添加通用数据 if (modelAndView != null) { modelAndView.addObject("serverTime", new Date()); } } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { // 记录异常信息 if (ex != null) { System.out.println("【异常】请求处理出错: " + ex.getMessage()); } // 清理 ThreadLocal 等资源 } }

注册 Interceptor:

@Configuration public class WebMvcConfig implements WebMvcConfigurer { @Autowired private PermissionInterceptor permissionInterceptor; @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(permissionInterceptor) .addPathPatterns("/admin/**", "/user/**") // 拦截路径 .excludePathPatterns("/login", "/public/**"); // 排除路径 } }

四、核心区别对比

对比维度

Filter

Interceptor

规范来源

Servlet 规范(J2EE)

Spring MVC 框架

框架依赖

不依赖 Spring,任何 Web 应用可用

必须在 Spring 环境中使用

拦截范围

所有请求(含静态资源、JSP)

仅 DispatcherServlet 映射的请求

执行时机

Servlet 之前

Controller 前后

方法数量

1 个 doFilter()

3 个:preHandlepostHandleafterCompletion

可获取对象

ServletRequestServletResponse

HttpServletRequestHandlerMethodModelAndView

异常处理

只能捕获 Filter 内部异常

afterCompletion可捕获 Controller 异常

静态资源拦截

支持

默认不支持


五、执行顺序详解

当 Filter 和 Interceptor 同时存在时,执行流程如下:

1. Filter.doFilter() 前置代码 ↓ 2. Interceptor.preHandle() ↓ 3. Controller 方法执行 ↓ 4. Interceptor.postHandle() ↓ 5. 视图渲染 ↓ 6. Interceptor.afterCompletion() ↓ 7. Filter.doFilter() 后置代码

示意图:

请求进入 ↓ [Filter 1] ──┐ ↓ │ [Filter 2] │ 过滤器链(双向拦截) ↓ │ [Servlet] │ ↓ │ [Interceptor 1.preHandle] ──┐ ↓ │ [Interceptor 2.preHandle] │ ↓ │ [Controller 执行] │ 拦截器链(三阶段) ↓ │ [Interceptor 2.postHandle] │ ↓ │ [Interceptor 1.postHandle] │ ↓ │ [视图渲染] │ ↓ │ [Interceptor 1.afterCompletion]┘ ↓ [Interceptor 2.afterCompletion] ↓ [Filter 2 后置] ↓ [Filter 1 后置] ↓ 响应返回

六、实战选择指南

6.1 优先使用 Filter 的场景

  • 统一字符编码:需要在请求进入 Servlet 前就设置编码
  • 跨域处理(CORS):处理 OPTIONS 预检请求,早于 Spring MVC
  • 安全过滤:XSS、SQL 注入的输入预处理
  • 全局日志:记录所有 HTTP 请求的完整生命周期
  • 非 Spring 环境:纯 Servlet 应用

6.2 优先使用 Interceptor 的场景

  • 登录状态校验:可便捷获取 HttpSession 和 Spring Bean
  • 细粒度权限控制:结合 @PreAuthorize、Security 上下文
  • 性能监控:精确统计 Controller 执行时间
  • 业务操作日志:获取 HandlerMethod 知道调用了哪个方法
  • 视图数据增强:在 postHandle 中向 Model 添加数据

七、黄金组合实践

在实际项目中,两者配合使用才能发挥最大价值:

层级

组件

职责

外层

Filter

编码设置、CORS、XSS 过滤、请求日志

内层

Interceptor

登录验证、权限校验、性能监控、业务日志

典型配置示例:

// Filter 处理通用底层逻辑 @Bean public FilterRegistrationBean<CharacterEncodingFilter> encodingFilter() { FilterRegistrationBean<CharacterEncodingFilter> bean = new FilterRegistrationBean<>(); bean.setFilter(new CharacterEncodingFilter("UTF-8", true)); bean.addUrlPatterns("/*"); bean.setOrder(Ordered.HIGHEST_PRECEDENCE); // 最先执行 return bean; } // Interceptor 处理业务逻辑 @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new AuthInterceptor()) .addPathPatterns("/api/**") .excludePathPatterns("/api/login"); }

八、面试高频问题

Q1:Filter 和 Interceptor 的本质区别是什么?

Filter 是 Servlet 规范,由容器管理;Interceptor 是 Spring MVC 机制,由框架管理。Filter 更早执行,能拦截所有请求;Interceptor 更晚执行,只能拦截 Spring MVC 请求,但能获取更丰富的上下文。

Q2:如何让 Interceptor 拦截静态资源?

修改 DispatcherServleturl-pattern/*,但会带来性能开销,通常不建议。静态资源拦截建议用 Filter。

Q3:afterCompletion 一定会执行吗?

只要 preHandle 返回 true 并执行了,无论后续是否异常,afterCompletion 都会执行,适合资源清理。

Q4:如何控制多个 Filter 的执行顺序?

使用 FilterRegistrationBeansetOrder() 方法,数字越小优先级越高。


九、总结

Filter

Interceptor

定位

容器层"护城河"

框架层"城门守卫"

优势

通用、底层、无框架依赖

精细、灵活、可获取 Spring 上下文

最佳实践

处理通用、底层问题

处理业务、安全、监控逻辑

核心原则:Filter 做"通用处理",Interceptor 做"业务拦截",两者协同构建完整的请求处理体系。

Read more

龙虾(OpenClaw)搭配本地千问模型(零token成本)实现电脑AI助理

龙虾(OpenClaw)搭配本地千问模型(零token成本)实现电脑AI助理

前言:现在AI助手遍地都是,但要么是云端服务要花token钱,要么是功能单一只能聊天,想找一个“不花钱、能干活、保隐私”的电脑AI助理,简直比登天!直到我发现了一个神仙组合——龙虾AI(OpenClaw)+ 本地千问模型,完美解决所有痛点:零token成本、全程本地运行、能接管电脑干活,无论是办公摸鱼还是高效产出,都能轻松拿捏。 本文是纯新手向原创实操教程,全程手把手,从工具认知、环境准备,到龙虾与本地千问的联动配置,再到实战场景演示,每一步都标清重点、避开坑点,不用懂复杂代码,不用花一分钱,普通人跟着走,10分钟就能拥有专属本地AI电脑助理,从此告别云端token焦虑和隐私泄露风险! 一、先搞懂:为什么是“龙虾+本地千问”?核心优势碾压同类组合 在开始操作前,先跟大家说清楚两个核心工具的作用,以及为什么它们搭配起来是“王炸”——毕竟市面上AI工具那么多,选对组合才能少走弯路,真正实现“零成本、高效率”。 1. 两个核心工具,

AI时代的技术民主化:为什么文科生可能成为最大受益者?

AI时代的技术民主化:为什么文科生可能成为最大受益者?

✨道路是曲折的,前途是光明的! 📝 专注C/C++、Linux编程与人工智能领域,分享学习笔记! 🌟 感谢各位小伙伴的长期陪伴与支持,欢迎文末添加好友一起交流! 当技术门槛被无限降低,真正有价值的不再是"怎么写代码",而是"想做什么" 01 一个被忽视的趋势 过去一年,我观察到一个有趣的现象:那些在AI浪潮中赚得盆满钵满的人,并不是技术背景最深厚的那批。 相反,他们中有学中文的、学设计的、学市场营销的。他们有一个共同特点——擅长理解人,擅长讲故事,擅长发现需求。 而这,恰恰是AI目前做不到的。 02 从"技术壁垒"到"创意壁垒" 传统开发流程 vs AI辅助流程 让我们看看传统的产品开发流程与现在的对比: 关键洞察:传统模式下,"想法&

告别AI代码“失忆症“!Claude Code效率翻倍的2个插件实战指南

告别AI代码"失忆症"!Claude Code效率翻倍的2个插件实战指南 引言:当AI变成"不靠谱队友"的那些糗事 想象一下,你刚给Claude Code布置完"加个博客评论区"的任务。第二天打开对话,他一脸懵地问:"你是说要给文章加个红色五角星吗?"这种"AI失忆症"是不是让你想摔键盘? 别慌!今天要分享的这套组合拳——Superpower工作流+Claude mem记忆插件,能让你的AI编程效率直接飙到300%,让"AI写代码如行云流水"不再是梦! 一、Superpower工作流:给AI装个"项目管理大脑" 1.1 传统开发VS Superpower开发,

AI世界模型(World Model)全解析:技术原理、研究进展与产业落地

AI世界模型(World Model)全解析:技术原理、研究进展与产业落地 摘要:世界模型(World Model)作为连接AI感知、决策与行动的核心枢纽,正成为突破通用人工智能(AGI)瓶颈的关键技术。本文从概念溯源、理论基础出发,系统剖析世界模型的技术架构、核心分类与实现方法,结合2024-2026年最新研究成果(如LeCun团队潜在动作世界模型、DIAMOND扩散模型)与产业落地案例,深入探讨其在强化学习、游戏开发、自动驾驶、机器人等领域的应用价值,最后梳理当前技术挑战并展望未来研究方向。全文兼顾学术深度与工程实践,为AI研究者与技术从业者提供全面的世界模型知识体系。 一、引言:从“符号拟合”到“世界理解”,AI的认知革命 1.1 大语言模型的认知瓶颈 自ChatGPT掀起大模型浪潮以来,大语言模型(LLM)凭借海量文本数据的统计拟合能力,在语义理解、内容生成、逻辑推理等领域展现出惊人实力。但在杨立昆、李飞飞等顶尖学者眼中,当前LLM仍是“