前言

学习官方文档,发现对于过滤器有分为三类

  • 默认过滤器
  • 自定义过滤
  • 全局过滤器

于是就有一个疑问,关于这些过滤器的访问顺序是怎样的,今天就以一个demo来进行测试

准备阶段

过滤器工厂类

以此为模板,复制出几份就可以了,注意打印信息,可区分就行

public class ExampleGatewayFilterFactory extends AbstractGatewayFilterFactory {

    /**
* 创造出的过滤器的顺序
*/
private int order; /**
* constructor
*/
public ExampleGatewayFilterFactory(int order) {
this.order = order;
} @Override
public GatewayFilter apply(Object config) {
return new InnerFilter();
} /**
* 创建一个内部类,来实现2个接口,指定顺序
*/
private class InnerFilter implements GatewayFilter, Ordered { @Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
System.out.println(" pre 自定义过滤器工厂 " + this.getClass().getSimpleName());
// 在then方法里的,相当于aop中的后置通知
return chain.filter(exchange).then(Mono.fromRunnable(() -> {
System.out.println(" post 自定义过滤器工厂 " + this.getClass().getSimpleName());
}));
} @Override
public int getOrder() {
return order;
}
} }

过滤器配置类

这里我们设置了

  • 2个默认过滤器
  • 2个自定义过滤器,
  • 1个全局过滤器
@SpringBootConfiguration
public class FilterConfig { // 以下是全局的过滤器(注意这里我们保持它为最高优先级)
@Bean
@Order(value = Ordered.HIGHEST_PRECEDENCE)
public GlobalFilter costFilter(){
return new CostFilter();
} // 以下是自定义的的过滤器工厂 @Bean
public GatewayFilterFactory exampleAGatewayFilterFactory(){
return new ExampleAGatewayFilterFactory(0);
} @Bean
public GatewayFilterFactory exampleGatewayFilterFactory(){
return new ExampleGatewayFilterFactory(1);
} // 以下是默认过滤工厂 @Bean
public GatewayFilterFactory myDefaultGatewayFilterFactory(){
return new MyDefaultGatewayFilterFactory(2);
} @Bean
public GatewayFilterFactory myDefaultAAGatewayFilterFactory(){
return new MyDefaultAAGatewayFilterFactory(3);
}
}

springboot启动类

// 这里扫描基础包,可以把其他配置加入进来
@SpringBootApplication(scanBasePackages = {"com.example.gateway"})
public class GatewayApplication { public static void main(String[] args) {
SpringApplication.run(GatewayApplication.class, args);
} }

application.yml文件的配置

spring:
application:
name: demo-gateway
cloud:
gateway:
discovery:
locator:
enabled: true
routes:
- id: to-route
uri: lb://demo-consumer
# 例如访问 localhost:8080/route会路由到demo-consumer的服务下的/route接口
predicates:
- Path=/route
- id: to-account
uri: lb://demo-consumer
predicates:
- Path=/account/v1/**
filters:
- name: ExampleA
- name: Example
# 默认过滤器
default-filters:
- name: MyDefaultAA
- name: MyDefault

测试

场景一

按照准备阶段的配置order顺序,来看访问的结果

  pre 自定义过滤器工厂 AAAA  InnerFilter
pre 自定义过滤器工厂 InnerFilter
pre 【默认】过滤器工厂 InnerFilter
pre 【默认AAA】过滤器工厂 InnerFilter
pre 全局过滤器 CostFilter
post 全局过滤器 CostFilter
post 【默认AAA】过滤器工厂 InnerFilter
post 【默认】过滤器工厂 InnerFilter
post 自定义过滤器工厂 InnerFilter
post 自定义过滤器工厂 AAAA InnerFilter

过滤器的执行顺序与堆栈这个数据结构很想,LIFO,gateway中的过滤器只有前置和后置2个生命周期,pre中后触发的,在post中就先被执行了

从输出的打印来看,默认过滤器和自定义的过滤器按照我们定义的1 2 3 4的顺序,成功的打印了出来,而全局过滤器是在最后才执行

场景二

自定义过滤器和默认过滤器都保留为相同的order顺序

修改如下,都统一为是0

    @Bean
public GatewayFilterFactory exampleAGatewayFilterFactory(){
return new ExampleAGatewayFilterFactory(0);
} @Bean
public GatewayFilterFactory exampleGatewayFilterFactory(){
return new ExampleGatewayFilterFactory(0);
} // 以下是默认过滤工厂 @Bean
public GatewayFilterFactory myDefaultGatewayFilterFactory(){
return new MyDefaultGatewayFilterFactory(0);
} @Bean
public GatewayFilterFactory myDefaultAAGatewayFilterFactory(){
return new MyDefaultAAGatewayFilterFactory(0);
}

返回的结果

  pre 【默认AAA】过滤器工厂  InnerFilter
pre 【默认】过滤器工厂 InnerFilter
pre 自定义过滤器工厂 AAAA InnerFilter
pre 自定义过滤器工厂 InnerFilter
pre 全局过滤器 CostFilter
post 全局过滤器 CostFilter
post 自定义过滤器工厂 InnerFilter
post 自定义过滤器工厂 AAAA InnerFilter
post 【默认】过滤器工厂 InnerFilter
post 【默认AAA】过滤器工厂 InnerFilter

注意到我们前面在yml中配置的过滤器的顺序是

          filters:
- name: ExampleA
- name: Example
# 默认过滤器
default-filters:
- name: MyDefaultAA
- name: MyDefault
  1. 这与前面几行的输出结果一致,所以如果当优先order一样的前提下,默认过滤器的执行优先于自定义过滤器,过滤器的执行顺序是与你在yml中声明的顺序是一致的。
  2. 全局过滤器依然是最后一个执行的

场景三

我们将自定义过滤器和默认过滤的顺序按照穿插的来,即

自定义、默认、自定义、默认

    // 以下是自定义的的过滤器工厂

    @Bean
public GatewayFilterFactory exampleAGatewayFilterFactory(){
return new ExampleAGatewayFilterFactory(1);
} @Bean
public GatewayFilterFactory exampleGatewayFilterFactory(){
return new ExampleGatewayFilterFactory(3);
} // 以下是默认过滤工厂 @Bean
public GatewayFilterFactory myDefaultGatewayFilterFactory(){
return new MyDefaultGatewayFilterFactory(2);
} @Bean
public GatewayFilterFactory myDefaultAAGatewayFilterFactory(){
return new MyDefaultAAGatewayFilterFactory(4);
}

运行访问后的执行结果如下

  pre 自定义过滤器工厂 AAAA  InnerFilter
pre 【默认】过滤器工厂 InnerFilter
pre 自定义过滤器工厂 InnerFilter
pre 【默认AAA】过滤器工厂 InnerFilter
pre 全局过滤器 CostFilter
post 全局过滤器 CostFilter
post 【默认AAA】过滤器工厂 InnerFilter
post 自定义过滤器工厂 InnerFilter
post 【默认】过滤器工厂 InnerFilter
post 自定义过滤器工厂 AAAA InnerFilter

可以看到它是按照我们书写的顺序来的


结论

  1. 全局过滤器与其他2类过滤器相比,永远是最后执行的;它的优先级只对其他全局过滤器起作用
  2. 当默认过滤器与自定义过滤器的优先级一样时,优先出发默认过滤器,然后才是自定义过滤器;同类型的过滤器,出发顺序与他们在配置文件中声明的顺序一致
  3. 默认过滤器与自定义过滤器使用同样的order顺序空间,即他们会按照各自的顺序来进行排序

spring boot gateway 过滤器的执行顺序的更多相关文章

  1. Spring Boot配置过滤器的两种方式

    过滤器(Filter)是Servlet中常用的技术,可以实现用户在访问某个目标资源之前,对访问的请求和响应进行拦截,常用的场景有登录校验.权限控制.敏感词过滤等,下面介绍下Spring Boot配置过 ...

  2. Spring Cloud Gateway过滤器精确控制异常返回(实战,完全定制返回body)

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 Spring Cloud Gateway应用 ...

  3. 解说asp.net core MVC 过滤器的执行顺序

    asp.net core MVC 过滤器会在请求管道的各个阶段触发.同一阶段又可以注册多个范围的过滤器,例如Global范围,controller范围等.以ActionFilter为例,我们来看看过滤 ...

  4. Spring Boot的属性加载顺序

        伴随着团队的不断壮大,往往不需要开发人员知道测试或者生产环境的全部配置细节,比如数据库密码,帐号信息等.而是希望由运维或者指定的人员去维护配置信息,那么如果要修改某项配置信息,就不得不去修改项 ...

  5. Spring Cloud Gateway过滤器精确控制异常返回(实战,控制http返回码和message字段)

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 前文<Spring Cloud Gat ...

  6. Spring Boot使用过滤器和拦截器分别实现REST接口简易安全认证

    本文通过一个简易安全认证示例的开发实践,理解过滤器和拦截器的工作原理. 很多文章都将过滤器(Filter).拦截器(Interceptor)和监听器(Listener)这三者和Spring关联起来讲解 ...

  7. spring boot 自定义过滤器链

    spring boot 会按照order值的大小,从大到小的顺序来依次过滤. 贴下代码: package com.osp.ucenter; import org.springframework.boo ...

  8. Spring Boot Maven 打包可执行Jar文件!

    Maven pom.xml 必须包含 <packaging>jar</packaging> <build> <plugins> <plugin&g ...

  9. spring boot mybatis 打成可执行jar包后启动UnsatisfiedDependencyException异常

    我的spring boot + mybatis项目在idea里面执行正常,但发布测试环境打成可执行jar包后就启动失败,提示错误如下: [ ERROR] [2018-08-30 17:23:48] o ...

随机推荐

  1. nginx转发上传图片接口图片的时候,报错413

    我这边有一个接口是上传图片,使用nginx进行代理,上传大一点的图片,直接调用我的接口不会报错,但是调用nginx上传图片就会报错"413 Request Entity Too Large& ...

  2. 使用git客户端免密码进行拉取等相关操作

    前言 如果使用git客户端进行pull或push操作时,遇到有权限的项目总要输入用户名密码,真的是太麻烦了,因此需要稍作修改,然后就可以免密码操作啦! 方法: 进入C盘->用户->你的主机 ...

  3. 教你30秒解开手机的密码 适用于高通CPU

    教程简介 先将手机进入9008模式.进入方法请自己百度. 进入9008方法如下:   先将手机关机,然后按住音量加和音量减不松手. 使用教程: 将数据线拆入电脑.会出现一个端口   出现端口后可以松开 ...

  4. Day5 - 05 函数的参数-关键字参数

    可变参数可以传入任意个参数,并在函数调用时自动组为一个tuple,而关键字参数允许传入任意个携带参数名的参数,这些关键字参数在函数内部自动组为一个dict.         >>> ...

  5. python 的基本语法

    python3 默认的编码格式 :utf-8 标识符: 1.可以是数字,字母,下划线组成 2.不能以数字开头 3.不能以保留字命名(保留字即是关键字,比如"print") 4.区分 ...

  6. mybatis-plus逻辑删除

    MP(mybatis plus)已经大大简化了我们好多的开发操作,基本的增删改查都有了,包括代码生成等等,今天想说的是它的逻辑删除功能.我们都在数据库设计时候经常会有is字段,表示是否删除,为了留下员 ...

  7. .NET生态系统掠影

    如果你是一名开发人员,想要进入到.NET的世界,你需要知道都有哪些可能.由于.NET Framework是..NET生态系统中最流行的技术,你可以用它来构建各种各样的应用程序,但是最近,出现了一些新的 ...

  8. 简单的冒泡排序算法(java)

    package lianxi; public class BubbleSort { public static void main(String[] args) { int[] array = {12 ...

  9. C#使用时间戳

    前言 参考博客 C#获取和转换时间戳: https://blog.csdn.net/weixin_39885282/article/details/79462443 获取时间戳: https://ww ...

  10. Ubuntu系统的ifconfig命令不能执行

    新安装的Ubuntu想要用WinSCP传文件时发现,ifconfig命令用不了 ping www.baidu.com 获得回应,应该是ifconfig未安装 解决这个问题,首先如图(时间较长,获取:[ ...