0

0

Java过滤器原理_Java Filter工作机制与生命周期原理

煙雲

煙雲

发布时间:2026-01-09 12:12:49

|

331人浏览过

|

来源于php中文网

原创

Filter 由 Servlet 容器在启动时扫描注册并按序调用 doFilter(),其生命周期由 init()、doFilter()、destroy() 三方法管控,且 request 流只能读一次,需用 HttpServletRequestWrapper 包装实现重放。

java过滤器原理_java filter工作机制与生命周期原理

Filter 是怎么被容器调用的?

Java Web 中的 Filter 不是靠你手动 new 或调用生效的,而是由 Servlet 容器(如 Tomcat)在启动时扫描、实例化,并按配置顺序插入请求处理链。它本质是一个“拦截器”,但和 Spring 的 HandlerInterceptor 不同——Filter 在 Servlet 生命周期之外,更底层,能拦请求头、请求体、响应体,甚至跳过目标 Servlet。

  • 容器启动时读取 web.xml 或注解(@WebFilter),注册 Filter 实例到内部过滤器链
  • 每次 HTTP 请求进来,容器按声明顺序依次调用 doFilter();只有显式调用 chain.doFilter(request, response),才会继续向后传递
  • Filter 实例是单例的(每个类一个实例),但 doFilter() 方法是多线程并发执行的,所以不能在 Filter 内部用非线程安全的成员变量存请求数据

init() / doFilter() / destroy() 什么时候触发?

Filter 的生命周期由容器完全管理,三个方法的触发时机非常明确,但容易误用:

  • init():容器启动完成、Filter 实例化后立即调用一次,且仅一次。适合做一次性初始化(如加载配置、初始化连接池)。注意:FilterConfig 可从中获取 init-param,但无法拿到 ServletContext 的完整引用(需通过 config.getServletContext() 获取)
  • doFilter():每次匹配请求都会调用,参数是包装过的 ServletRequest/ServletResponse(可能是 HttpServletRequest/HttpServletResponse,但必须向下转型才可用 HTTP 特有方法)
  • destroy():容器关闭前调用一次,用于释放资源。但不要指望它一定被执行(比如 kill -9 进程就跳过)

为什么 request.getInputStream() 只能读一次?Filter 里怎么安全重放?

这是 Filter 最常踩的坑:HTTP 请求体(如 JSON)默认是流式读取,request.getInputStream()request.getReader() 调用一次后流就关闭了,后续 Servlet 拿不到原始数据。

雪鸮AI
雪鸮AI

高效便捷的智能绘图辅助工具,一键生成高质量效果图。

下载
  • 解决方案不是“缓存字节”,而是用装饰器模式包装请求对象,例如继承 HttpServletRequestWrapper
  • 关键点:在 doFilter() 开头就把整个 body 读入内存(IOUtils.toString(request.getInputStream(), "UTF-8")),再构造一个可重复读的 HttpServletRequestWrapper 子类,重写 getInputStream()getReader() 返回新缓存
  • ⚠️ 注意:大文件上传场景下别这么干,会 OOM;生产环境建议只对特定路径(如 /api/**)或小 payload(Content-Length )启用缓存
public class BodyCachingRequestWrapper extends HttpServletRequestWrapper {
    private final byte[] cachedBody;

    public BodyCachingRequestWrapper(HttpServletRequest request) throws IOException {
        super(request);
        this.cachedBody = IOUtils.toByteArray(request.getInputStream());
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {
        return new CachedServletInputStream(cachedBody);
    }

    // ... 省略 getReader() 实现
}

@WebFilter 和 web.xml 配置哪个优先?

两者可以共存,但容器加载顺序有明确规定:web.xml 中定义的 Filter 总是先于 @WebFilter 注册。也就是说,如果你在 web.xml 里配了 A、B,在代码里用 @WebFilter 声明了 C,最终执行顺序是 A → B → C(前提是 URL 匹配规则一致)。

立即学习Java免费学习笔记(深入)”;

  • @WebFilter(urlPatterns = "/api/*")/api/* 效果等价,但注解无法设置 类型(如 ERROR、FORWARD),除非用 dispatcherTypes 属性显式指定
  • Tomcat 9+ 默认禁用 web.xml 的自动扫描(如果用了 metadata-complete="true"),此时 @WebFilter 是唯一方式
  • Spring Boot 项目默认不加载 web.xml,所有 Filter 必须用 @Bean 方式注册或 @WebFilter + @ServletComponentScan
Filter 的真正复杂点不在写法,而在它介入的是容器最底层的 I/O 流与线程模型。一个没考虑线程安全的成员变量、一次未 reset 的输入流、一个忘了 chain.doFilter() 的 if 分支,都可能让整个请求链静默失败。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

829

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

735

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

3

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.4万人学习

C# 教程
C# 教程

共94课时 | 6.4万人学习

Java 教程
Java 教程

共578课时 | 44.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号