Spring Boot过滤器链:从入门到精通

文章目录

    • 一、过滤器链是什么?
    • 二、为什么需要过滤器链?
    • 三、Spring Boot中的过滤器链是如何工作的?
      • (一)过滤器的生命周期
      • (二)过滤器链的执行流程
    • 四、如何在Spring Boot中定义自己的过滤器?
    • 五、如何控制过滤器的顺序?
    • 六、Spring Security中的过滤器链
    • 七、过滤器链的性能优化
    • 八、总结

在当今的Web开发中,Spring Boot凭借其简洁高效的特性,成为了众多开发者的首选框架。而过滤器链作为Spring Boot中一个重要的概念,对于保障应用的安全性和功能完整性起着关键作用。今天,就让我们一起深入了解Spring Boot中的过滤器链,从入门到精通,一步步揭开它的神秘面纱。

一、过滤器链是什么?

在Spring Boot中,过滤器链(Filter Chain)是由一系列过滤器(Filter)组成的有序集合。这些过滤器按照一定的顺序依次对请求进行处理,每个过滤器都可以对请求进行拦截、修改或增强操作,从而实现诸如安全检查、日志记录、请求转发等功能。过滤器链的设计使得我们可以将不同的功能模块化,通过组合多个过滤器来实现复杂的功能逻辑,同时保持代码的清晰和可维护性。

二、为什么需要过滤器链?

在Web应用中,我们常常需要在请求到达业务逻辑之前或之后执行一些通用的操作。例如,在请求到达控制器之前,我们可能需要验证用户的身份、记录请求的日志;在响应返回给客户端之前,我们可能需要对响应内容进行压缩或加密。如果这些操作都直接写在业务逻辑代码中,会导致代码的耦合度很高,难以维护和扩展。而过滤器链的出现,正是为了解决这个问题。通过将这些通用操作封装到不同的过滤器中,并按照一定的顺序组织成过滤器链,我们可以在不修改业务逻辑代码的情况下,灵活地添加或修改这些通用操作,从而提高代码的可维护性和可扩展性。

三、Spring Boot中的过滤器链是如何工作的?

(一)过滤器的生命周期

在Spring Boot中,每个过滤器都遵循一个标准的生命周期,主要包括以下三个阶段:

  • 初始化(init):当过滤器被创建时,Spring会调用过滤器的init方法。在这个方法中,我们可以进行一些初始化操作,例如加载配置文件、初始化资源等。不过,在Spring Boot中,由于过滤器通常是通过Spring的依赖注入机制进行管理的,因此我们很少需要手动实现init方法。
  • 过滤(doFilter):这是过滤器的核心方法,当请求到达过滤器时,Spring会调用doFilter方法。在这个方法中,我们可以对请求进行拦截、修改或增强操作。例如,我们可以在doFilter方法中验证用户的身份,如果用户未登录,则直接返回401状态码;如果用户已登录,则继续调用下一个过滤器。doFilter方法的参数包括ServletRequest、ServletResponse和FilterChain,其中FilterChain表示过滤器链的下一个过滤器,通过调用FilterChain的Filterdo方法,可以将请求传递给下一个过滤器。
  • 销毁(destroy):当过滤器被销毁时,Spring会调用过滤器的destroy方法。在这个方法中,我们可以进行一些清理操作,例如关闭资源、释放内存等。和init方法一样,在Spring Boot中,我们也很少需要手动实现destroy方法。

(二)过滤器链的执行流程

当一个请求到达Spring Boot应用时,Spring会按照过滤器链中过滤器的顺序依次调用每个过滤器的doFilter方法。每个过滤器都可以对请求进行处理,然后通过调用FilterChain的doFilter方法将请求传递给下一个过滤器。如果某个过滤器不想让请求继续传递下去,它可以选择不调用FilterChain的doFilter方法,从而直接返回响应给客户端。例如,如果一个过滤器发现用户未登录,它可以直接返回401状态码,而不再调用下一个过滤器。
过滤器链的执行流程可以用以下伪代码来表示:

filterChain.doFilter(ServletRequest request, ServletResponse response) {for (Filter filter : filters) {filter.doFilter(request, response, new FilterChain() {@Overridepublic void doFilter(ServletRequest request, ServletResponse response) {// 调用下一个过滤器FilterChain.doFilter(request, response);}});}
}

从这个伪代码中可以看出,过滤器链的执行是一个递归的过程,每个过滤器都负责调用下一个过滤器的doFilter方法,直到所有的过滤器都执行完毕,请求才会最终到达业务逻辑代码。

四、如何在Spring Boot中定义自己的过滤器?

在Spring Boot中,定义自己的过滤器非常简单。我们只需要创建一个类,让它实现javax.servlet.Filter接口,然后在类上添加@Component注解,让Spring能够自动扫描并注册这个过滤器。以下是一个简单的自定义过滤器示例:

import org.springframework.stereotype.Component;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;@Component
public class MyFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {// 初始化操作System.out.println("MyFilter初始化");}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {// 在请求到达业务逻辑之前执行的操作System.out.println("MyFilter请求前处理");HttpServletRequest httpRequest = (HttpServletRequest) request;System.out.println("请求路径:" + httpRequest.getRequestURI());// 调用下一个过滤器chain.doFilter(request, response);// 在响应返回给客户端之前执行的操作System.out.println("MyFilter响应后处理");}@Overridepublic void destroy() {// 销毁操作System.out.println("MyFilter销毁");}
}

在上面的代码中,我们定义了一个名为MyFilter的过滤器。在doFilter方法中,我们在请求到达业务逻辑之前打印了一条日志,并获取了请求的路径;在响应返回给客户端之前,我们又打印了一条日志。通过在类上添加@Component注解,Spring会自动扫描并注册这个过滤器,将其加入到过滤器链中。

五、如何控制过滤器的顺序?

在Spring Boot中,过滤器链中过滤器的执行顺序是由过滤器的优先级决定的。默认情况下,Spring会按照过滤器的类名的字典顺序对过滤器进行排序。但是,我们可以通过实现Ordered接口或使用@Order注解来指定过滤器的优先级。优先级越小,过滤器越先执行。
以下是一个使用@Order注解指定过滤器优先级的示例:

import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;import javax.servlet.*;
import.io java.IOException;@Component
@Order(1)
public class FirstFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {System.out.println("FirstFilter初始化");}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {System.out.println("FirstFilter请求前处理");chain.doFilter(request, response);System.out.println("First响应Filter后处理");}@Overridepublic void destroy() {System.out.println("FirstFilter销毁");}
}@Component
@Order(2)
public class SecondFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {System.out.println("SecondFilter初始化");}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {System.out.println("SecondFilter请求前处理");chain.doFilter(request, response);System.out.println("SecondFilter响应后处理");}@Overridepublic void destroy() {System.out.println("SecondFilter销毁");}
}

在上面的代码中,我们定义了两个过滤器FirstFilter和SecondFilter,并通过@Order注解分别指定了它们的优先级为1和2。因此,在过滤链器中,FirstFilter会先于SecondFilter执行。

六、Spring Security中的过滤器链

Spring Security是Spring Boot中用于实现安全认证和授权的框架,它也使用了过滤器链来实现各种安全功能。Spring Security的过滤器链中包含了许多预定义的过滤器,例如SecurityContextPersistenceFilter用于在请求开始时恢复安全上下文,在请求结束时清理安全上下文;UsernamePasswordAuthenticationFilter用于处理基于用户名和密码的登录请求;ExceptionTranslationFilter用于处理安全相关的异常,例如用户未登录或没有权限访问某个资源等。
通过自定义过滤器并将其加入到Spring Security的过滤器链中,我们可以扩展Spring Security的功能,实现自己的安全需求。例如,我们可以通过自定义一个过滤器来实现基于JWT(JSON Web Token)的认证机制,或者通过自定义一个过滤器来实现对请求的访问控制。
以下是一个将自定义过滤器加入到Spring Security过滤器链中的示例:

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity;
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;@Configuration
@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {@Overrideprotected void configure(HttpSecurity http) throws Exception {http.addFilterBefore(new MyCustomFilter(), UsernamePasswordAuthenticationFilter.class).authorizeRequests().anyRequest().authenticated().and().formLogin().permitAll().and().logout().permitAll();}
}

在的上面代码中,我们通过addFilterBefore方法将自定义的MyCustomFilter加入到了Spring Security的过滤器链中,并指定它在UsernamePasswordAuthenticationFilter之前执行。这样,我们就可以在用户登录之前对请求进行处理,例如验证请求的来源是否合法等。

七、过滤器链的性能优化

虽然过滤器链为我们提供了强大的功能,但是过多的过滤器或不合理的过滤器链设计可能会对应用的性能产生影响。以下是一些优化过滤器链性能的建议:
减少不必要的过滤器:只在真正需要的地方使用过滤器,避免在每个请求上都执行不必要的操作。例如,如果某个过滤器只对特定的请求路径有效,可以通过在doFilter方法中添加路径匹配逻辑来减少不必要的调用。
优化过滤器的实现:在过滤器的doFilter方法中,尽量减少对资源的占用和对性能的影响。例如,避免在过滤器中进行复杂的计算或大量的I/O操作。
合理安排过滤器的顺序:将那些可以快速拒绝请求的过滤器放在前面,例如安全过滤器或权限过滤器,这样可以减少不必要的后续处理。
使用异步过滤器:如果过滤器的操作可以异步执行,可以考虑使用AsyncFilter来提高性能。异步过滤器可以在不阻塞主线程的情况下执行耗时操作,从而提高应用的并发能力。

八、总结

通过本文的介绍,我们从过滤器链的基本概念出发,逐步深入到了Spring Boot中过滤器链的实现原理、定义方法、顺序控制以及与Spring Security的结合等方面。过滤器链作为一种强大的功能,可以帮助我们实现各种通用的操作,提高代码的可维护性和可扩展性。然而,在使用过滤器链时,我们也需要注意性能优化,避免对应用的性能产生负面影响。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/70940.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【博客之星】GIS老矣尚能饭否?WebGIS项目实战经验与成果展示

目录 一、最前面的话 二、前言 1、关于“夜郎king” 3、GIS的“老骥伏枥” 4、WebGIS的“新程启航” 三、WebGIS技术简介 1、前、后技术简介 2、系统功能架构 四、WebGIS项目应用效果 1、应急灾害 2、交通运输 3、智慧文旅 4、其它项目 五、未来与展望 1、云计算…

C++中什么时候用. 什么时候用->

学了一年C今天出了一个大岔子,因为太久没有做链表类型题目了,并且STL用惯了今天遇到一题,写的时候发现完全不对劲,搞慌了,首先我们看题目 2. 两数相加 再看我第一次的解答,先不论结果对不对 错的行为有很多…

本地部署DeepSeek Nodejs版

目录 1.下载 Ollama 2.下载DeepSeek模型 3.下载 ollama.js 1.下载 Ollama https://ollama.com/ 下载之后点击安装,等待安装成功后,打开cmd窗口,输入以下指令: ollama -v 如果显示了版本号,则代表已经下载成功了。…

java项目之足球联赛管理系统源码(ssm+mysql)

风定落花生,歌声逐流水,大家好我是风歌,混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的足球联赛管理系统。项目源码以及部署相关请联系风歌,文末附上联系信息 。 项目简介: 足球联赛管理系统的主要使用者…

【C】链表算法题7 -- 环形链表||

leetcode链接https://leetcode.cn/problems/linked-list-cycle-ii/description/ 问题描述 给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。如果链表中有某个节点,可以通过连续跟踪 next 指针再次到…

uniapp开发h5部署到服务器

1.发行>网站-PC Web或手机H5(仅适用于uniapp) 2.填写网站域名 3.编译成功后会生成一个unpackage文件夹找到下面的h5 4.接下来会使用一个工具把h5里面的文件放到服务器上面(WinSCP使用其他能部署的工具也行) 5.登录 6.登录成功后…

亚信安全正式接入DeepSeek

亚信安全致力于“数据驱动、AI原生”战略,早在2024年5月,推出了“信立方”安全大模型、安全MaaS平台和一系列安全智能体,为网络安全运营、网络安全检测提供AI技术能力。自2024年12月DeepSeek-V3发布以来,亚信安全人工智能实验室利…

小白零基础如何搭建CNN

1.卷积层 在PyTorch中针对卷积操作的对象和使用的场景不同,如有1维卷积、2维卷积、 3维卷积与转置卷积(可以简单理解为卷积操作的逆操作),但它们的使用方法比较相似,都可以从torch.nn模块中调用,需要调用的…

21vue3实战-----git husky和git commit规范

21vue3实战-----git husky和git commit规范 1.husky工具1.1目的1.2如何做到这一点?1.3步骤 2.git commit规范2.1使用Commitizen自动生成规范格式供选择2.2代码提交风格2.3代码提交验证 之前在https://blog.csdn.net/fageaaa/article/details/145474065文章中已经讲了在vue项目…

arduino扩展:Arduino Mega 控制 32 个舵机(参考表情机器人)

参考:表情机器人中使用22个舵机的案例 引言 在电子制作与自动化控制领域,Arduino 凭借其易用性和强大的扩展性备受青睐。Arduino Mega 作为其中功能较为强大的一款开发板,具备丰富的引脚资源,能够实现复杂的控制任务。舵机作为常…

PyQt学习记录03——批量设置水印

0. 目录 PyQt学习记录01——加法计算器 PyQt学习记录02——串口助手 1. 前言 本次主要是为了学习Qt中的 QFileDialog 函数, QFileDialog.getExistingDirectory:用于选择文件夹,返回的是一个文件夹路径。 QFileDialog.getOpenFileName&…

Visual Studio 使用 “Ctrl + /”键设置注释和取消注释

问题:在默认的Visual Studio中,选择单行代码后,按下Ctrl /键会将代码注释掉,但再次按下Ctrl /键时,会进行双重注释,这不是我们想要的。 实现效果:当按下Ctrl /键会将代码注释掉,…

社区版IDEA中配置TomCat(详细版)

文章目录 1、下载Smart TomCat2、配置TomCat3、运行代码 1、下载Smart TomCat 由于小编的是社区版,没有自带的tomcat server,所以在设置的插件里面搜索,安装第一个(注意:安装时一定要关闭外网,小编因为这个…

Flink-DataStream API

一、什么样的数据可以用于流式传输 Flink的DataStream API 允许流式传输他们可以序列化的任何内容。Flink自己的序列化程序用于 基本类型:即字符串、长、整数、布尔值、数组复合类型:元组、POJO和Scala样例类 基本类型我们已经很熟悉了,下…

渗透利器:Burp Suite 联动 XRAY 图形化工具.(主动扫描+被动扫描)

Burp Suite 联动 XRAY 图形化工具.(主动扫描被动扫描) Burp Suite 和 Xray 联合使用,能够将 Burp 的强大流量拦截与修改功能,与 Xray 的高效漏洞检测能力相结合,实现更全面、高效的网络安全测试,同时提升漏…

Java 大视界 -- 深入剖析 Java 在大数据内存管理中的优化策略(49)

💖💖💖亲爱的朋友们,热烈欢迎你们来到 青云交的博客!能与你们在此邂逅,我满心欢喜,深感无比荣幸。在这个瞬息万变的时代,我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

JVM ②-双亲委派模型 || 垃圾回收GC

这里是Themberfue 在上节课对内存区域划分以及类加载的过程有了简单的了解后,我们再了解其他两个较为重要的机制,这些都是面试中常考的知识点,有必要的话建议背出来,当然不是死记硬背,而是要有理解的背~~~如果对 JVM …

html文件怎么转换成pdf文件,2025最新教程

将HTML文件转换成PDF文件,可以采取以下几种方法: 一、使用浏览器内置功能 打开HTML文件:在Chrome、Firefox、IE等浏览器中打开需要转换的HTML文件。打印对话框:按下CtrlP(Windows)或CommandP(M…

2025 西湖论剑wp

web Rank-l 打开题目环境: 发现一个输入框,看一下他是用上面语言写的 发现是python,很容易想到ssti 密码随便输,发现没有回显 但是输入其他字符会报错 确定为ssti注入 开始构造payload, {{(lipsum|attr(‘global…

Web前端开发--HTML

HTML快速入门 1.新建文本文件&#xff0c;后缀名改为.html 2.编写 HTML结构标签 3.在<body>中填写内容 HTML结构标签 特点 1.HTML标签中不区分大小写 2.HTML标签属性值中可以使用单引号也可使用双引号 3.HTML语法结构比较松散&#xff08;但在编写时要严格一点&…