《Spring Cloud Gateway 快速入门:从路由到自定义 Filter 的完整教程》​

article/2025/6/24 1:16:32

1.网关介绍

在前面的学习中,我们通过Eureka和Nacos解决了辅助注册,使用Spring Cloud LoadBalance解决了负载均衡的问题,使用OpenFeign解决了远程调用的问题。

但是当前的所有微服务的接口都是直接对外暴露的,外部是可以直接访问的,为了保证对外服务的安全性,服务端实现的微服务接口通常都带有一定的权限校验机制,由于我们使用了微服务,原本一个应用被拆成了多个模块,所以程序员不得不多次实现校验逻辑,当这套校验逻辑需要修改时,程序员需要修改多个应用,加重了开发人员的负担

针对以上问题,一个常用的解决方案是使用API网关

API网关也简称为网关,网关通常也是单独一个服务,通常是后端的唯一入口,网关的设计模式主要是门面模式,网关就是类似于整个微服务的架构,所有的外部客户端访问后端服务时,都必须要经过网关来进行调度和过滤

网关的核心功能:

1.权限控制: 作为微服务的入口,对用户的权限进行校验,如果校验失败则进行拦截

2.动态路由:一切请求先经过网关,但网关不进行业务处理,而是根据某种规则,将请求妆发到某一个微服务

3.负载均衡:当路由的目标服务有多个时,还需要进行负载均衡

4.限流:请求流量过高时,按照网关中的配置对微服务能够接受的流量进行放行,避免服务压力过大

2.常见的网关实现

业界常用的网关方式众多,技术方案成熟,开源产品丰富,如Nginx、Kong、Zuul、Spring Cloud Gateway等。以下介绍两种常见网关方案:

  • ​Zuul​​:Netflix开源的API网关组件,是Spring Cloud Netflix子项目核心组件之一,可与Eureka、Ribbon、Hystrix等配合使用。在Spring Cloud Finchley正式版前,Spring Cloud推荐使用Netflix的Zuul(指Zuul 1.X)。但Netflix在2018年宣布部分组件进入维护状态,不再开发新特性,其中包含Zuul。
  • ​Spring Cloud Gateway​​:Spring Cloud全新的API网关项目,基于Spring + Spring Boot等技术开发,旨在替代Zuul,为微服务架构提供简单有效的请求转发途径,并提供安全性、监控/指标和弹性等横切关注点。 

3.Spring Cloud Gateway的快速上手 

注意:这里我是基于我前面的代码来使用的,各位读者视自己的情况而定 

1.创建一个网关服务

2.引入网关依赖 

由于网关设计到Nacos上的服务,负载均衡,所以还要额外将这两个依赖引入网关工程的pom文件中,所以,在网关工程中的pom文件引入下面的依赖

        <!--网关依赖--><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-gateway</artifactId></dependency><!--基于nacos实现服务发现依赖--><dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId></dependency><!--实现负载均衡的依赖--><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-loadbalancer</artifactId></dependency>

3.编写启动类 

@SpringBootApplication
public class GatewayApplication {public static void main(String[] args) {SpringApplication.run(GatewayApplication.class,args);}
}

4.添加Gateway的路由配置 

 创建appication.yml文件,添加响应的配置,添加的配置有网关的端口号,Nacos服务发现的配置项和网关对应的配置项,如一下配置

server:port: 10030   #网关端口号,随便配置一个即可,只要端口号不冲突
spring:application:name: gatewaycloud:nacos:discovery:server-addr: 8.134.130.95:10020 #Nacos服务发现的配置gateway:routes:- id: orders-service   #路由规则id, 随便起, 不重复即可uri: lb://orders-service/ #目标服务地址predicates:   #路由条件- Path=/order/**,/feign/**- id: product-serviceuri: lb://product-service/predicates:- Path=/product/**

 配置字段说明:

1. id:自定义路由,随便起,不重复即可

2. uri:目标服务地址,支持普通uri和 lb://应用注册的服务名称  ,其中lb表示负载均衡,lb:// 方式表示从注册中心获取的服务地址

3.predicate:路由条件,根据匹配结果决定是否将请求转发到对应的微服务中,上述代码中,我们将符合Path规则的一切请求都有网关转发到uri参数指定的地址(也就是uri指定的注册在Nacos上的微服务) 

5.测试网关

启动网关服务,并重启订单服务和产品服务

1.通过网关访问product-service服务,url和predicates的Path条件匹配成功,则成功访问product-service服务

不匹配,则会报404,如下图

2.通过网关访问订单服务,url匹配成功的话,就会成功访问订单服务

3.访问订单服务中的其他服务,成功访问

4.Predicate 

Predicate是Java8提供的一个函数式编程接口,它用于接收一个参数并返回一个布尔值,用于请求校验,请求参数的校验,Prdicate里面的核心方法就是test()方法,如下面截图

1.代码演示---判断一个字符串是否是空字符串 

1.定义一个Predicate,通过实现Predicate接口

/*** 判断字符串是否为空* true ->空字符串* false -> 非空字符串*/
public class StringPredicate implements Predicate<String> {@Overridepublic boolean test(String s) {return s==null||s.isEmpty();}}

2.编写测试方法:

    @Testpublic void Test(){Predicate predicate=new StringPredicate();System.out.println(predicate.test(""));System.out.println(predicate.test("aa"));}

运行截图:

2.Predicate的其他写法

1.匿名内部类写法 

代码演示:

    /*** 匿名内部类*/@Testpublic void Test2(){Predicate<String> predicate=new Predicate<String>() {@Overridepublic boolean test(String s) {return s==null||s.isEmpty();}};System.out.println(predicate.test(""));System.out.println(predicate.test("aa"));}

运行截图

2.lambda表达式写法 

代码演示:

    /*** lambda表达式*/@Testpublic void Test3(){Predicate<String> predicate= s -> s==null||s.isEmpty();System.out.println(predicate.test(""));System.out.println(predicate.test("aa"));}

运行截图:

3.Predicate中的其他方法 

1.negate()方法

negate() 是用来对Predicate返回结果取反的方法,如下图

代码演示:

    /*** 取反->negate()*/@Testpublic void test(){Predicate<String> predicate= s -> s==null||s.isEmpty();System.out.println(predicate.negate().test(""));System.out.println(predicate.negate().test("aa"));}

运行截图:

2.and()方法 

and()方法其实就是条件A&&条件B,即当前调用and()方法的predicate的test方法&&other的test方法 

代码演示:

    /*** and* 字符串不为空,且字符串有数字组成  "aa","bb"*/@Testpublic void test2(){Predicate<String> predicate= s-> s==null||!s.isEmpty();Predicate<String> predicate1=s -> s.chars().allMatch(Character::isDigit);System.out.println(predicate.and(predicate1).test("12"));System.out.println(predicate.and(predicate1).test(""));System.out.println(predicate.and(predicate1).test("a1"));}

运行截图:

3.or() 

or()方法其实就是条件A||条件B,即当前调用or()方法的predicate的test方法||other的test方法  

代码演示:

    /*** or* 判断字符串是不是aa或者bb*/@Testpublic void test3(){Predicate<String> predicate1=s-> s.isEmpty();Predicate<String> predicate=s -> s.equals("aa");Predicate<String> predicate2=s -> s.equals("bb");System.out.println(predicate2.or(predicate1).test("bb"));System.out.println(predicate2.or(predicate1).test("cc"));}

运行截图:

5. Route Predicate Factories

Route Predicate Factories(路由断言工厂),在Spring Cloud Gateway中,Predicate提供了路由规则的匹配机制 

在配置文件中写的断言规则只是字符串,这些字符串会被Route Predicate Factories读取并处理,转变为路由判断的条件

Spring Cloud Gateway默认提供了很多Route  Predicate Factory,这些Predicate会分别匹配HTTP请求中的不同属性,并且多个Predicate会默认为and逻辑组合

1.通过时间匹配 

Predicate支持设置一个时间,在请求转发的时候,可以通过判断是否在这个时间之前转发或者之后转发。比如现在我设置只有在2025年5月29号时,该请求才能转发请求,在这之前,不会将请求转发,我就可以添加一下配置:

Spring是通过ZoneDateTimela来对时间进行的对比,ZoneDateTime是Java8中日期时间功能里,用于表示带时区的日期与时间信息的类,ZoneDateTime支持通过时区来设置时间 

 添加完以上配置,重启网关服务,去调用订单服务,发现服务调用成功

如果添加一个Before配置,如下图

此时,重启网关服务,此时就会报404,因为此时访问时间已经在Before的条件的时间之后 

还有很多匹配规则,可以去Spring官网查看,这里就不一 一说明了 

6. Gateway Filter  Factories(网关过滤器工厂)

Predicate决定了请求由哪一个路由处理,如果需要在请求处理前后添加一些逻辑,我们此时就需要用到Filter(过滤器)

Filter分为两种类型:Pre类型和Post类型 

Pre类型过滤器:路由处理之前执行(请求转发到后端服务之前执行),在Pre类型过滤器中可以做身份验证和限流登操作

Post类型过滤器:请求执行完后完成,也就是将结果返回给客户端之前执行 

Spring Cloud Gateway中内置了很多FIlter,用于拦截和链式处理web请求,比如权限校验,访问超时等设定

Spring Cloud Gateway从作用范围上,可以将Filter分为GatewayFilterGlobalFilter 

GatewayFilter:应用到单个路由或者一个分组的路由上

GlobalFilter:应用到所有的路由上,也就是对所有的请求生效

6.1 GatewayFilter

GateFilter同Predicate类似,都是在配置文件中配置,每个过滤器的逻辑都是固定的

1.为当前请求添加参数---AddRequestParameter,添加下面的配置

2. 为当前请求添加Header---AddRequestHeader

3.从当前请求中删除某一个Header信息---RemoveRequestHeader

4.为响应结果添加Header---AddResponseHeader

过程图:

5.从响应结果中删除某个Header---RemoveResponseHeader 

此时通过Postman发现,p6方法中添加的Header信息已经被删除 

6.默认过滤器---default-filters 

添加一个filter并将其应用到所有路由,并配置了重传次数 

订单服务

产品服务

6.2 GlobalFilter 

GlobalFilter是Spring Cloud Gateway中的全局过滤器,它和Gateway的作用是相同的,但是GlobalFilter会应用到所有的路由请求上,全局过滤器通常用于实现与安全性,性能监控和日志记录等相关的全局功能

Spring Cloud Gateway内置的全局过滤器有很多,比如:

1.Gateway Metrics Filter:网关指标,提供监控指标

2.Forward Routing Filter:用于本地forword,请求不转发到下游服务器

3.LoadBalancer Client Filter:针对下游服务,实现负载均衡

1.快速上手---以Gateway Metrics Filter为例 

在网关工程中添加下面的依赖:

        <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-actuator</artifactId></dependency>

然后在网关工程中的配置文件中添加相关的配置,如下图

spring:cloud:gateway:metrics:enabled: true
management:endpoints:web:exposure:include: "*"endpoint:health:show-details: alwaysshutdown:enabled: true

重启网关工程,通过下面的链接去访问,将会显示所有监控的信息

链接:127.0.0.1:10030/actuator 

2.过滤器的执行顺序 

一个项目中,既有GatewayFilter,又有GlobalFilter,执行的先后顺序是什么呢?

请求路由后,网关会把当前项目中的GatewayFilter和GlobalFilter合并到一个过滤器链中,并进行排序,依次执行过滤器  

每一个过滤器都必须指定一个int类型的order值,默认值为0,表示过滤的优先级,order值越小,优先级越高,执行顺序越靠前。 

1、Filter通过实现Order接口或者天剑@Order注解来指定Order值

2、Spring Cloud Gateway提供的Filter有Spring指定,用户也可以自定义Filter,由用户指定

3、当过滤器的Oorder值一样,会按照dafaultFilter>GatewayFlter>ClobalFilter的顺序来执行 

7.自定义过滤器 

Spring Cloud Gateway提供了过滤器的扩展功能,开发者可以根据实际业务来自定义过滤器,同样自定义过滤器也支持GatewayFilter和GlobalFilter两种 

1.自定义GatewayFilter 

自定义GatewayFilter需要去实现对应的接口GatewayFilterFactory,SpringBoot默认帮我们实现的抽象类是AbstractGatewayFilterFactory,可以直接使用,由于我们还要声明该过滤器的优先级,所以还要去实现Ordered接口,分别去重写里面的方法

其次,对于过滤器来说,所有的路由配置都是通过配置文件来实现的,配置文件中写的是一些字符串,在应用启动时,Gateway会将这些字符串解析为内部数据结构,当请求达到网关时,Gateway会从解析后的配置提取参数,并传给对应的Filter实现类,所以要实现一个自定义GAtewayFilter,我们还要创建一个对应类型的数据类,作为自定义Gateway的参数类型

完整实现:

对应的参数类---CustomConfig

@Data
public class CustomConfig {private String name;
}

自定义GAtewayFilter的实现代码---CustomGatewayFilterFactory

@Slf4j
@Component
public class CustomGatewayFilterFactory extends AbstractGatewayFilterFactory<CustomConfig> implements Ordered {public CustomGatewayFilterFactory() {super(CustomConfig.class);}@Overridepublic GatewayFilter apply(CustomConfig config) {return new GatewayFilter() {@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {//Pre -> 执行请求 -> Postlog.info("Pre Filter,config:{}",config);//pre类型过滤器代码逻辑return chain.filter(exchange).then(Mono.fromRunnable(()->{log.info("Post Filter,config:{}",config);}));}};}@Overridepublic int getOrder() {return Ordered.LOWEST_PRECEDENCE;}
}

代码解析:

1.类名统一以GatewayFilter结尾,因为默认情况下,过滤器的name会采用该定义类的前缀,以上面的自定义GateawayFilter为例,此时自定义GateawayFilter的name就是Custom

2.apply方法中同时包含Pre和Post,then方法就是Post类型的逻辑

3.CustomConfig是一个配置类,也yml配置对应

4.CustomGatewayFilter需要交给Spring管理,所以需要添加@Component注解

5.getOder表示该过滤器的优先级,值越大,优先级越低

最后去配置文件中添加相关配置

测试:

2.自定义GlobalFilter 

GlobalFilter的实现比较简单,它不需要额外的配置,只需要实现GlobalFilter接口即可,自动会过滤所有的Filter

代码实现:

@Slf4j
@Component
public class CustomGlobalFilter implements GlobalFilter,Ordered {@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {log.info("Global Filter Pre...");return chain.filter(exchange).then(Mono.fromRunnable(()->{log.info("Global Filter Post...");}));}@Overridepublic int getOrder() {return Ordered.LOWEST_PRECEDENCE;}
}

测试:


http://www.hkcw.cn/article/TIbuRoyoWj.shtml

相关文章

高并发抽奖系统优化方案

引子 最近接触了一个抽奖的项目&#xff0c;由于用户量比较大&#xff0c;而且第三方提供的认证接口并发量有限&#xff0c;为了保证服务的高可用性&#xff0c;所以对高并限制发有一定的要求。经过一系列研究和讨论&#xff0c;做出了以下一些优化方案。 需求分析 根据用户量…

【Doris基础】Apache Doris中的Segment详解:存储与查询的核心组件

目录 1 Segment概述 1.1 什么是Segment 1.2 Segment的重要性 2 Segment的物理结构 2.1 Segment文件组成 2.2 列数据存储格式 3 Segment的生命周期 3.1 Segment创建 3.2 Segment合并(Compaction) 3.3 Segment过期与清理 4 Segment与查询处理 4.1 查询中的Segment剪枝…

JAVA中的枚举

代码示例 理论 枚举的行为多态 public enum MyLockStrategy {SKIP_FAST(){Overridepublic boolean tryLock(RLock lock, MyLock prop) throws InterruptedException {return lock.tryLock(0, prop.leaseTime(), prop.unit());}},FAIL_AFTER_RETRY_TIMEOUT(){Overridepublic bo…

电脑驱动程序更新工具, 3DP Chip 中文绿色版,一键更新驱动!

介绍 3DP Chip 是一款免费的驱动程序更新工具&#xff0c;可以帮助用户快速、方便地识别和更新计算机硬件驱动程序。 驱动程序更新工具下载 https://pan.quark.cn/s/98895d47f57c 软件截图 软件特点 简单易用&#xff1a;用户界面简洁明了&#xff0c;操作方便&#xff0c;…

day61—DFS—省份数量(LeetCode-547)

题目描述 有 n 个城市&#xff0c;其中一些彼此相连&#xff0c;另一些没有相连。如果城市 a 与城市 b 直接相连&#xff0c;且城市 b 与城市 c 直接相连&#xff0c;那么城市 a 与城市 c 间接相连。 省份 是一组直接或间接相连的城市&#xff0c;组内不含其他没有相连的城市…

MySql(十)

目录 准备工作 1&#xff09;准备一张表 2&#xff09;插入数据 1.排序 1--asc 升序 2--desc 降序 3--组合排序 2.聚合函数 1.count(字段名) 2.sum(字段名) 3.max(字段名) 4.min(字段名) 5.avg(字段名) 准备工作 1&#xff09;准备一张表 CREATE table role(roleid INT PRIMAR…

LabVIEW Val (Sgnl) 属性

在 LabVIEW 事件驱动架构中&#xff0c;Val (Sgnl) 属性&#xff08;Value (Signaling)&#xff09;是实现编程触发与用户交互行为一致性的关键技术。与普通 Value 属性不同&#xff0c;Val (Sgnl) 在修改控件值的同时强制生成值改变事件&#xff0c;确保程序逻辑与 UI 交互保持…

解常微分方程组

Euler法 function euler_method % 参数设置 v_missile 450; % 导弹速度 km/h v_enemy 90; % 敌艇速度 km/h % 初始条件 x0 0; % 导弹初始位置 x y0 0; % 导弹初始位置 y xe0 120; % 敌艇初始位置 y t0 0; % 初始时间 % 时间步长和总时间 dt 0.01; % 时间步长 t_final …

「Java教案」数据类型、变量与常量

课程目标 1&#xff0e;知识目标 能够根据Java基本数据类型的分类、存储规则及适用场景&#xff0c;合理的选择数据类型。能在合适的场景下正确声明和定义变量和常量。能够根据显式和隐式数据类型转换的规则与风险&#xff0c;合理的进行数据类型转换。 2&#xff0e;能力目…

本地部署基于 Kibana 开源搜索引擎 Elasticsearch 并实现外部访问

Elasticsearch 是一个分布式、高扩展、高实时的搜索与数据分析引擎。它能很方便的使大量数据具有搜索、分析和探索的能力。Kibana 是一个开源可视化工具&#xff0c;用于与 Elasticsearch 进行集成&#xff0c;提供大量数据分析。 本文将详细的介绍如何利用 Docker 在本地部署…

【Ollama】windows部署ollama并运行模型

一、Ollama安装 1.下载Ollama 官网&#xff1a;https://ollama.com/ 2.安装 点击下载 安装完成后打开cmd窗口 键盘按住WinR输入cmd 3.Ollama常用指令 指令说明ollama list查看本地已下载的模型列表ollama pull <模型名>下载模型&#xff08;如 ollama pull llama…

Linux入门(十二)服务管理

服务本质就是进程&#xff0c;但是在后台运行&#xff0c;通常会监听某个端口&#xff0c;等等其他的程序来访问 systemctl 管理指令 systemctl [start | stop | restart | reload | status ] systemctl status NetworkManagersystemctl 服务是在/usr/lib/systemd/system 查看 …

在Ubuntu20.04上安装ROS Noetic

本章教程,主要记录在Ubuntu20.04上安装ROS Noetic。 一、添加软件源 sudo sh -c . /etc/lsb-release && echo "deb http://mirrors.tuna.tsinghua.edu.cn/ros/ubuntu/ `lsb_release -cs` main" > /etc/apt/sources.list.d/ros-latest.list二、设置秘钥 …

Linux 库制作与原理

文章目录 1. 库的概念2. 静态库2.1 静态库的制作2.2 静态库的原理 3. 动态库的制作4.ELF文件4.1 ELF文件内容4.2 ELF文件链接与加载 5. ELF与进程地址空间&#xff0c;动静态链接和库5.1 ELF与静态链接5.2 ELF与进程地址空间5.3 ELF与动态链接、动态库5.3.2 动态库5.3.2 动态链…

26考研——文件管理_文件目录(4)

408答疑 文章目录 二、文件目录1、目录的作用与结构1.1、目录的基本概念1.2、目录的组织形式1.2.1、单级目录结构1.2.2、两级目录结构1.2.3、多级&#xff08;树形&#xff09;目录结构1.2.4、无环图目录结构 1.3、目录的实现方式1.3.1、线性列表1.3.2、哈希表 2、文件共享与链…

俄军操作系统 Astra Linux 安装教程

安装 U盘制作 Rufus 写盘工具&#xff1a;https://rufus.ie/ Astra Linux ISO 镜像文件&#xff1a;https://dl.astralinux.ru/astra/stable/2.12_x86-64/iso/ 准备一个8g以上的u盘&#xff0c;打开Rufus写盘工具&#xff0c;选择下载的iso镜像&#xff0c;写入u盘&#xff…

MacOS安装Docker Desktop并汉化

1. 安装Docker Desktop 到Docker Desktop For Mac下载对应系统的Docker Desktop 安装包&#xff0c;下载后安装&#xff0c;没有账号需要注册&#xff0c;然后登陆即可。 2. 汉化 前往汉化包下载链接下载对应系统的.asar文件 然后将安装好的文件覆盖原先的文件app.asar文件…

AI 的早期萌芽?用 Swift 演绎约翰·康威的「生命游戏」

文章目录 摘要描述题解答案题解代码分析示例测试及结果时间复杂度空间复杂度总结 摘要 你有没有想过&#xff0c;能不能通过简单的规则模拟出生与死亡&#xff1f;「生命游戏」正是这样一种充满魅力的数学模拟系统。这篇文章我们来聊聊它的规则到底有多神奇&#xff0c;并用 S…

系统设计——系统架构分层设计经验

摘要 文章通过对 MVC 三层架构和 DDD 四层架构的深入分析&#xff0c;结合具体的代码示例和项目结构设计&#xff0c;为 Java 开发人员提供了全面的系统架构分层设计经验。在实际项目中&#xff0c;开发人员应根据项目的规模和业务复杂度选择合适的架构模式&#xff0c;并遵循…

Windows下编译zlib

本文记录在Windows下编译zlib的流程。 零、环境 操作系统Windows 11VS Code1.92.1Git2.34.1Visual StudioVisual Studio Community 2022CMake3.22.1 一、编译 1.1 下载代码 git clone https://github.com/madler/zlib.git 1.2 构建 按照下表配置CMake&#xff0c; CMake…