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

news2025/6/2 19:05:23

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: gateway
  cloud:
    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-service
          uri: 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> {

    @Override
    public boolean test(String s) {
        return s==null||s.isEmpty();
    }
    
}

2.编写测试方法:

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

运行截图:

2.Predicate的其他写法

1.匿名内部类写法 

代码演示:

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

运行截图

2.lambda表达式写法 

代码演示:

    /**
     * lambda表达式
     */
    @Test
    public 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()
     */
    @Test
    public 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"
     */
    @Test
    public 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
     */
    @Test
    public 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: always
    shutdown:
      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);
    }

    @Override
    public GatewayFilter apply(CustomConfig config) {
        return new GatewayFilter() {
            @Override
            public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
                //Pre -> 执行请求 -> Post
                log.info("Pre Filter,config:{}",config);//pre类型过滤器代码逻辑
                return chain.filter(exchange).then(Mono.fromRunnable(()->{
                    log.info("Post Filter,config:{}",config);
                }));
            }
        };
    }

    @Override
    public 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 {

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        log.info("Global Filter Pre...");
        return chain.filter(exchange).then(Mono.fromRunnable(()->{
            log.info("Global Filter Post...");
        }));
    }

    @Override
    public int getOrder() {
        return Ordered.LOWEST_PRECEDENCE;
    }
}

测试:

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

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

相关文章

第3节 Node.js 创建第一个应用

Node.js 非常强大&#xff0c;只需动手写几行代码就可以构建出整个HTTP服务器。事实上&#xff0c;我们的Web应用以及对应的Web服务器基本上是一样的。 在我们创建Node.js第一个"Hello, World!"应用前&#xff0c;让我们先了解下Node.js应用是由哪几部分组成的&…

我们来学mysql -- “数据备份还原”sh脚本

数据备份&还原 说明执行db_backup_cover.sh脚本 说明 环境准备&#xff1a;来源数据库(服务器A)&#xff1b;目标数据库(服务器B)dbInfo.sh脚本记录基本信息 来源库、目标库的ip、port及执行路径 # MySQL 客户端和 mysqldump 的路径 MYSQL_CLIENT"/work/oracle/mysql…

【排序算法】快速排序详解--附详细流程代码

快速排序算法 介绍 快速排序&#xff08;Quick Sort&#xff09;是一种高效的分治排序算法&#xff0c;由英国计算机科学家 Tony Hoare 于 1960 年提出。它是实际应用中最常用的排序算法之一。快速排序的基本思想是&#xff1a;选择一个"基准"&#xff08;pivot&am…

解决各个系统报错TDengine:no taos in java.library.path问题

windows 系统解决办法 在本地上安装一个TD的Windows客户端&#xff0c;注意安装的客户端版本一定要和服务端TD版本完全一致。&#xff08;或者将 C:\TDengine\driver\taos.dll 拷贝到 C:\Windows\System32\ 目录下&#xff09; 客户端各个历史版本下载链接&#xff1a;TDengin…

java helloWord java程序运行机制 用idea创建一个java项目 标识符 关键字 数据类型 字节

HelloWord public class Hello{public static void main(String[] args) {System.out.print("Hello,World!");} }java程序运行机制 用idea创建一个java项目 建立一个空项目 新建一个module 注释 标识符 关键字 标识符注意点 数据类型 public class Demo02 {public st…

免费文本转语音工具体验:祈风TTS使用

简介&#xff1a;语音生成的另一种方式 现在很多人通过视频记录生活&#xff0c;表达观点。拍摄剪辑不难&#xff0c;配音成了常见难题。部分人对自己的声音不够自信&#xff0c;也有人在特定场景下不便出声。文本转语音工具可以成为解决方案。 常见的TTS&#xff08;Text To…

JS和TS的区别

JavaScript 与 TypeScript 的主要区别和特性对比 1. 基础定义 JavaScript 是一种动态、弱类型的编程语言&#xff0c;广泛应用于前端开发以及通过 Node.js 扩展到后端开发。TypeScript 则是 JavaScript 的超集&#xff0c;它在 JavaScript 的基础上添加了静态类型系统和其他增…

Python实现P-PSO优化算法优化BP神经网络分类模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 随着人工智能技术的快速发展&#xff0c;神经网络在分类任务中展现了强大的性能。BP&#xff08;Back Propagation&…

Linux --进度条小程序更新

这里使用随机数来模拟下载量&#xff0c;来实现一个下载进度更新的小程序 main.c 的代码&#xff0c;其中downlod这个函数使用的是函数指针&#xff0c;如果有多个进度条函数可以传入进行多样化的格式下载显示&#xff0c;还需要传入一个下载总量&#xff0c;每次"下载以…

关于镜像如何装进虚拟机

本篇文章为感谢小仙猪老师特别编写 本篇文章仅以Ubuntu为例 目录 创建虚拟机 汉化 如果没有China选项 检查网络 创建虚拟机 第一步&#xff0c;创建虚拟机 因为&#xff0c;第一个选项是会把虚拟机的文件放在c盘因此&#xff0c;这里博主选择自定义&#xff0c;然后下一…

智慧体育馆数字孪生,场馆管理智能化

图扑数字孪生智慧体育馆可视化管理平台。通过高精度三维建模&#xff0c;对体育馆建筑结构、设施设备等进行 1:1 虚拟映射&#xff0c;全方位还原场馆物理实体。系统集成多维度传感器数据&#xff0c;实现对人流量、客流密度、区域拥堵指数等信息的实时采集与分析&#xff0c;动…

回归算法模型之线性回归

哈喽&#xff01;我是 我不是小upper&#xff5e; 今天来和大家聊聊「线性回归」—— 这是机器学习里最基础、最直观的算法之一&#xff0c;咱们用一个超简单的例子就能搞懂它&#xff01; 先看一个生活场景 假设你是房产中介&#xff0c;遇到一个灵魂拷问&#xff1a; 客户有…

【深度学习】10. 深度推理(含链式法则详解)RNN, LSTM, GRU,VQA

深度推理&#xff08;含链式法则详解&#xff09;RNN, LSTM, GRU&#xff0c;VQA RNN 输入表示方式 在循环神经网络&#xff08;Recurrent Neural Network, RNN&#xff09;中&#xff0c;我们处理的是一段文字或语音等序列数据。对于文本任务&#xff0c;输入通常是单词序列…

【Qt】Bug:findChildren找不到控件

使用正确的父对象调用 findChildren&#xff1a;不要在布局对象上调用 findChildren&#xff0c;而应该在布局所在的窗口或控件上调用。

【linux】linux进程概念(四)(环境变量)超详细版

小编个人主页详情<—请点击 小编个人gitee代码仓库<—请点击 linux系列专栏<—请点击 倘若命中无此运&#xff0c;孤身亦可登昆仑&#xff0c;送给屏幕面前的读者朋友们和小编自己! 目录 前言一、基本概念二、认识常见的几个环境变量echo $ 查看某个环境变量env 显示…

从零开始的二三维CAD|CAE软件: 解决VTK,DICOM体素化-失效问题.

背景: 在从零开始的二三维软件开发中, 需要加载CT的dicoms影像文件, 并将其序列化之后的数据,体素化 可惜..vtk的c#库,将其体素化的时候,竟然失败... 使用vtkDicomReader ,设置 Dicom文件夹读取,竟然不停的失败...从网上找了一些版本.也没啥可用的资料... 解决办法: 直接…

【计算机网络】应用层协议Http——构建Http服务服务器

&#x1f525;个人主页&#x1f525;&#xff1a;孤寂大仙V &#x1f308;收录专栏&#x1f308;&#xff1a;计算机网络 &#x1f339;往期回顾&#x1f339;&#xff1a; 【Linux笔记】——进程间关系与守护进程 &#x1f516;流水不争&#xff0c;争的是滔滔不息 一、Http协…

linux版本vmware修改ubuntu虚拟机为桥接模式

1、先打开linux版本vmware操作界面 2、设置虚拟路由编辑器的桥接模式 输入账号密码 自动模式 不需要进行任何操作 3、修改虚拟机设置网络模式为桥接模式 然后save保存一下配置 4、现在进入虚拟机查看ens33配置 网卡启动但是没有ip 5、自己进行设置修改ubuntu网络配置文件 cd …

从0到1上手Trae:开启AI编程新时代

摘要&#xff1a;字节跳动 2025 年 1 月 19 日发布的 Trae 是一款 AI 原生集成开发环境工具&#xff0c;3 月 3 日国内版推出。它具备 AI 问答、代码自动补全、基于 Agent 编程等功能&#xff0c;能自动化开发任务&#xff0c;实现端到端开发。核心功能包括智能代码生成与补全、…

Linux之MySQL安装篇

1.确保Yum环境是否能正常使用 使用yum环境进行软件的安装 yum -y install mysql-server mysql2.确保软件包已正常完成安装 3.设置防火墙和selinux配置 ## 关闭防火墙 systemctl stop firewalld## 修该selinux配置 vim /etc/selinux/config 将seliuxenforcing修改为sel…