Spring Boot 框架下的过滤器

1. 过滤器的基础概念

1.1 什么是过滤器?
  • Servlet 规范的一部分,定义在 javax.servlet.Filter 接口中。
  • 在 HTTP 请求到达目标资源(如 Controller)之前或响应返回客户端之前,拦截并对其进行预处理或后处理。
1.2 过滤器的核心方法

javax.servlet.Filter 接口定义了以下三个方法:

  • init(FilterConfig filterConfig)

    • 在过滤器初始化时调用,通常用于加载资源或初始化配置。
  • doFilter(ServletRequest request, ServletResponse response, FilterChain chain)

    • 核心方法,用于拦截和处理请求或响应。
    • 调用 chain.doFilter(request, response) 表示将请求传递给下一个过滤器或目标资源。
  • destroy()

    • 在过滤器销毁时调用,用于释放资源。

2. Spring Boot 项目中如何使用过滤器

在 Spring Boot 中,可以通过以下两种方式注册过滤器:

2.1 自动注册过滤器

通过 @Component 注解将过滤器注册为 Spring Bean,Spring Boot 会自动加载并将其添加到过滤器链中。

示例代码:

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) {System.out.println("过滤器初始化");}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {HttpServletRequest httpRequest = (HttpServletRequest) request;System.out.println("请求路径:" + httpRequest.getRequestURI());chain.doFilter(request, response); // 继续传递请求}@Overridepublic void destroy() {System.out.println("过滤器销毁");}
}

解释:

  • @Component
    • 将该类注册为 Spring Bean,Spring Boot 自动将其作为过滤器加载。
    • 默认拦截所有请求(/*)。
  • doFilter 方法
    • HttpServletRequest 用于获取请求的详细信息(如路径、参数)。
    • chain.doFilter 表示继续将请求传递给下一个过滤器或目标资源。

2.2 手动注册过滤器

通过 FilterRegistrationBean 手动注册过滤器,可以指定 URL 匹配规则和执行顺序。

示例代码:

import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class FilterConfig {@Beanpublic FilterRegistrationBean<MyFilter> registerMyFilter() {FilterRegistrationBean<MyFilter> registrationBean = new FilterRegistrationBean<>();registrationBean.setFilter(new MyFilter()); // 注册过滤器registrationBean.addUrlPatterns("/api/*");  // 仅拦截 /api/ 开头的请求registrationBean.setOrder(1);              // 设置执行顺序,值越小优先级越高return registrationBean;}
}

解释:

  • FilterRegistrationBean
    • 用于手动注册过滤器,提供灵活的配置选项。
  • addUrlPatterns
    • 定义过滤器的作用范围,比如这里仅拦截 /api/* 的请求。
  • setOrder
    • 定义过滤器的执行顺序,值越小优先级越高。

3. 使用 Spring 提供的扩展过滤器

Spring 提供了一个扩展类 OncePerRequestFilter,它是对标准过滤器的增强,用于确保在一次请求中只执行一次过滤逻辑(防止重复过滤)。

示例代码:

import org.springframework.web.filter.OncePerRequestFilter;import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;public class MyOncePerRequestFilter extends OncePerRequestFilter {@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)throws ServletException, IOException {System.out.println("请求路径:" + request.getRequestURI());filterChain.doFilter(request, response); // 继续传递请求}
}

注册方式: 可以通过 @ComponentFilterRegistrationBean 注册,方式与标准过滤器一致。

解释:

  • OncePerRequestFilter
    • 确保在一次请求中只执行一次过滤器逻辑。
  • doFilterInternal
    • 处理请求的核心方法,逻辑和标准 FilterdoFilter 类似。

4. 过滤器的执行顺序 

在 Spring Boot 中,多个过滤器的执行顺序由以下规则决定: 

4.1 自动注册的过滤器

示例:

@Component
@Order(1) // 优先级最高
public class FirstFilter implements Filter {...
}@Component
@Order(2) // 优先级次之
public class SecondFilter implements Filter {...
}
  • 默认按加载顺序执行。
  • 可以通过 @Order 注解设置优先级(值越小优先级越高)。
4.2 手动注册的过滤器

示例:

registrationBean.setOrder(1);
  • 使用 FilterRegistrationBean#setOrder 方法设置优先级。

5. 常见过滤器应用场景

  • 用户认证和授权

    • 检查请求头中是否携带有效的 Token。
    • 验证用户权限。
  • 请求日志记录

    示例:

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {HttpServletRequest httpRequest = (HttpServletRequest) request;System.out.println("请求路径:" + httpRequest.getRequestURI());chain.doFilter(request, response);
    }
    
    • 记录请求的详细信息(如路径、IP 地址、参数)。
  • 跨域处理

    示例:

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {HttpServletResponse httpResponse = (HttpServletResponse) response;httpResponse.setHeader("Access-Control-Allow-Origin", "*");httpResponse.setHeader("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE");chain.doFilter(request, response);
    }
    
    • 在响应头中添加跨域支持信息。
  • 加解密

    • 对请求体进行解密,对响应体进行加密。

6. 注意事项

  • 避免业务逻辑的复杂化

    • 过滤器应专注于全局性逻辑(如日志、认证),复杂的业务逻辑应放在 Service 层处理。
  • 避免重复过滤

    • 使用 OncePerRequestFilter 防止在同一个请求中多次执行过滤器逻辑。
  • 路径匹配规则

    • 通过 addUrlPatterns 或在 doFilter 中手动判断,确保过滤器只作用于必要的路径。
  • 资源拦截控制

    • 排除静态资源(如 .css, .js)的拦截,避免对静态资源应用不必要的逻辑。

7.一个请求在整个Spring Boot Web 应用中的处理流程

以一个常见的 Spring Boot Web 应用为例,处理一个请求可能会经过多个环节:

[客户端请求] ↓
[Filter1 -> Filter2 -> ... -> FilterN] // 过滤器链↓
[Servlet Container] // DispatcherServlet↓
[Spring MVC 拦截器(Interceptor)] // 可选↓
[Controller / RestController 方法]↓
[视图解析或返回响应]↑
[FilterN -> ... -> Filter1] // 响应回程,也可以在过滤器里做后置处理↑
[客户端接收响应]
  • 过滤器链先于 Spring MVC 的拦截器执行。
  • Filter.doFilter 中,如果调用 chain.doFilter(request, response),请求才会进入下一个过滤器或目标 Servlet。
  • 响应返回时,也会逆向经过过滤器链,为我们提供后置处理的机会。

 

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

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

相关文章

多云架构,JuiceFS 如何实现一致性与低延迟的数据分发

随着大模型的普及&#xff0c;GPU 算力成为稀缺资源&#xff0c;单一数据中心或云区域的 GPU 资源常常难以满足用户的全面需求。同时&#xff0c;跨地域团队的协作需求也推动了企业在不同云平台之间调度数据和计算任务。多云架构正逐渐成为一种趋势&#xff0c;然而该架构下的数…

【Git原理和使用】Git 分支管理(创建、切换、合并、删除、bug分支)

一、理解分支 我们可以把分支理解为一个分身&#xff0c;这个分身是与我们的主身是相互独立的&#xff0c;比如我们的主身在这个月学C&#xff0c;而分身在这个月学java&#xff0c;在一个月以后我们让分身与主身融合&#xff0c;这样主身在一个月内既学会了C&#xff0c;也学…

静态路由配置与调试——计算机网络实训day1

文章目录 操作前准备一、实验目的二、实验要求三、实验过程1、在R1和R2上配置设备名称。基本配置设备命名 2、在R1和R2上配置接口IP地址&#xff0c;并查看IP地址的配置情况。3、在R1和R2上配置静态路由&#xff0c;并查看路由表。静态路由缺省路由&#xff08;默认路由&#x…

农产品直播带货方案拆解

作为一名经验丰富的营销策划人道叔&#xff0c;今天我来拆解一下咱们4A营销广告圈的这份《直播天府川农好物带货方案》&#xff0c;让你能学到很多实用的策略和技巧&#xff0c;直接应用到你的策划工作中去。 首先&#xff0c;咱们看看背景分析。 助农直播现在可是个大热门&a…

【Qt】控件概述和QWidget核心属性1(enabled、geometry、windowTitle、windowIcon、QRC机制)

一、控件概念 界面上各种元素、各种部分的统称&#xff08;如按钮、输入框、下拉框、单选复选框...&#xff09; Qt作为GUI开发框架&#xff0c;内置了各种的常用控件&#xff0c;并支持自定义控件。 二、控件体系发展 1.没有完全的控件&#xff0c;需要使用绘图API手动绘制…

【杂谈】-50+个生成式人工智能面试问题(四)

7、生成式AI面试问题与微调相关 Q23. LLMs中的微调是什么&#xff1f; 答案&#xff1a;虽然预训练语言模型非常强大&#xff0c;但它们并不是任何特定任务的专家。它们可能对语言有惊人的理解能力&#xff0c;但仍需要一些LLMs微调过程&#xff0c;开发者通过这个过程提升它…

spring boot 多数据源集成mysql、postgresql、phoenix、doris等

如何搭建多数据源项目只要以下简单几步; 一. 创建核心在config.datasource文件夹里 二. 引入相对应的jar包 三. 创建数据库连接配置 四. 写逻辑代码进行验证 1.DataSource package com.irootech.config.datasource;import java.lang.annotation.*;Target({ElementType.MET…

01-51单片机LED与独立按键

一、单片机概述 注意&#xff1a;个人学习笔记&#xff0c;里面涉及到的C语言和进程转换相关的知识在C语言部分已经写了&#xff0c;这里是默认都会的状态学习单片机。 1.什么是单片机 单片机&#xff0c;英文Micro Controller Unit&#xff0c;简称MCU。其内部集成了CPU、R…

音视频入门基础:MPEG2-PS专题(6)——FFmpeg源码中,获取PS流的视频信息的实现

音视频入门基础&#xff1a;MPEG2-PS专题系列文章&#xff1a; 音视频入门基础&#xff1a;MPEG2-PS专题&#xff08;1&#xff09;——MPEG2-PS官方文档下载 音视频入门基础&#xff1a;MPEG2-PS专题&#xff08;2&#xff09;——使用FFmpeg命令生成ps文件 音视频入门基础…

【网络协议】静态路由详解

网络中的路由器通过以下两种方式之一发现远程网络&#xff1a; 静态配置路由动态路由协议 在本文&#xff0c;我们将学习关于静态路由的各种概念&#xff0c;例如如何配置静态路由、路由表如何进行决策、路由接口等相关知识。 文章目录 引言直连网络静态路由路由表原则原则1原…

Dependency check 通过Maven构建时,配置Mysql数据库遇到的三个坑

使用过Dependency check的同学&#xff0c;一定会遇到这个问题—— 每次执行依赖扫描时&#xff0c;由于网络问题会导致NVD下载种子数据的过程中的种种失败&#xff0c;不仅浪费了大量时间&#xff0c;还会因为下载文件的不完整性直接导致依赖检测的失败。所以我在使用Dependen…

【面试题】技术场景 5、日志采集ELK

日志采集的重要性与采集方式 重要性&#xff1a;在项目开发、测试及生产环境中&#xff0c;日志是定位系统问题的关键手段&#xff0c;对系统维护与问题排查至关重要。采集方式 常规采集&#xff1a;按天保存日志文件至专门目录&#xff0c;文件名包含项目名、端口及日期&…

【数据库】三、SQL语言

文章目录 三、SQL语言1 概述2 数据定义(DDL)2.1 定义数据库2.2 定义基本表2.3 修改基本表2.4 删除基本表 3 数据操作(DML)3.1 数据查询3.1.1 单表查询3.1.2 连接查询3.1.3 嵌套查询3.1.4 集合查询 3.2 数据更新3.2.1 插入数据3.2.2 修改数据3.2.3 删除数据 4 数据控制(DCL)5 视…

Unity中 Xlua使用整理(二)

1.Xlua的配置应用 xLua所有的配置都支持三种方式&#xff1a;打标签&#xff1b;静态列表&#xff1b;动态列表。配置要求&#xff1a; 列表方式均必须是static的字段/属性 列表方式均必须放到一个static类 建议不用标签方式 建议列表方式配置放Editor目录&#xff08;如果是H…

Python Matplotlib教程-Matplotlib 多子图布局

Python Matplotlib 多子图布局 Matplotlib 是 Python 中最常用的数据可视化库&#xff0c;它提供了强大的功能来绘制不同类型的图表。在实际应用中&#xff0c;通常需要将多个图表绘制在同一个画布上&#xff0c;这就需要用到 多子图布局。本篇文章将详细介绍如何使用 Matplot…

全方位解读消息队列:原理、优势、实例与实践要点

全方位解读消息队列&#xff1a;原理、优势、实例与实践要点 一、消息队列基础认知 在数字化转型浪潮下&#xff0c;分布式系统架构愈发复杂&#xff0c;消息队列成为其中关键一环。不妨把消息队列想象成一个超级“信息驿站”&#xff0c;在古代&#xff0c;各地的信件、物资运…

Photon最新版本PUN 2.29 PREE,在无网的局域网下,无法连接自己搭建的本地服务器

1.图1为官方解答 2.就是加上这一段段代码&#xff1a;PhotonNetwork.NetworkingClient.SerializationProtocol SerializationProtocol.GpBinaryV16; 完美解决 unity 商店最新PUN 2 插件 不能连接 &#xff08;环境为&#xff1a;本地局域网 无外网情况 &#xff09; …

消息中间件类型介绍

消息中间件是一种在分布式系统中用于实现消息传递的软件架构模式。它能够在不同的系统或应用之间异步地传输数据&#xff0c;实现系统的解耦、提高系统的可扩展性和可靠性。以下是几种常见的消息中间件类型及其介绍&#xff1a; 1.RabbitMQ 特点&#xff1a; • 基于AMQP&#…

51单片机(二)中断系统与外部中断实验

中断即单片机因为某些原因E暂定现在的工作P0&#xff0c;转去做其他的工作P1&#xff0c;完了之后继续之前的事P0&#xff0c;其他工作P1就是中断程序&#xff0c;原因E就是中断事件&#xff0c;原因由外部发生&#xff0c;程序不能预测到的是硬中断&#xff0c;可以由程度触发…