Spring Cloud Gateway:GlobalFilter和GatewayFilter的区别与联系

. global filter和gateway filter区别
1.1 接口定义不一样
global filter和gateway filter是分别定义了一个接口,他们的结构看起来很像,下面是
glogal filter的定义:

// GlobalFilter自己就是一个单独的接口,没有任何继承关系,里面就一个方法filter。简单理解英文:
//  GlobalFilter是一个拦截样式,链路方式处理web请求,用来实现横切面,以及一些安全,超时等需求
/**
 * Contract for interception-style, chained processing of Web requests that may be used to
 * implement cross-cutting, application-agnostic requirements such as security, timeouts,
 * and others.
*/
public interface GlobalFilter {
    Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain);
}

gateway filter的定义如下,可以看到他们的接口描述基本一致,说明其用途也是相类似的。不一样的地方是GatewayFilter多定义两个属性,暂时没看到它的用途是什么。还有其继承了ShortcutConfigurable,这个应该是方便GatewayFilter做配置化的,因为GatewayFilter没有具体的实现类定义,必须从配置中读取filter信息,然后动态创建GatewayFilter。

/**
 * Contract for interception-style, chained processing of Web requests that may be used to
 * implement cross-cutting, application-agnostic requirements such as security, timeouts,
 * and others. Specific to a Gateway
 * 
 * Copied from WebFilter
 */
public interface GatewayFilter extends ShortcutConfigurable {
    String NAME_KEY = "name";
    String VALUE_KEY = "value";
    Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain);
}

1.2 实现类定义不一样
global filter有具体的类定义,我们可以在spring-cloud-gateway-server项目中找到对应的类定义,10个glogal filter类路径如下:

比如NettyWriteResponseFilter的定义:


gateway filter没有类定义,它只有工厂类,比如AddRequestHeaderGatewayFilterFactory,它可以在apply方法中创建匿名GatewayFilter:

public class AddRequestHeaderGatewayFilterFactory
        extends AbstractNameValueGatewayFilterFactory {

    @Override
    // 这个方法会返回一个GatewayFilter,可以猜测动态创建GatewayFilter时会调用这个方法
    public GatewayFilter apply(NameValueConfig config) {
        return new GatewayFilter() {
            @Override
            // 内部类实现filter逻辑
            public Mono<Void> filter(ServerWebExchange exchange,
                    GatewayFilterChain chain) {
                // 拿到head的值
                String value = ServerWebExchangeUtils.expand(exchange, config.getValue());
                // 设置head到请求里
                ServerHttpRequest request = exchange.getRequest().mutate()
                        .header(config.getName(), value).build();
                // 处理filter链
                return chain.filter(exchange.mutate().request(request).build());
            }
        };
    }
}


1.3 bean的加载方式不一样
global filter有具体的实现类,可以直接实例化加载到容器中,拿NettyRoutingFilter的加载为例,可以在GatewayAutoConfiguration中看到它是直接new出来放到容器中的。gateway filter没有具体的实现类,只有对应的工厂类,它的加载方式肯定就是不一样的,它是动态加载,暂时没研究是如何加载的,后面补上。

        // 全局filter的加载
        @Bean
        @ConditionalOnEnabledGlobalFilter
        public NettyRoutingFilter routingFilter(HttpClient httpClient,
                ObjectProvider<List<HttpHeadersFilter>> headersFilters,
                HttpClientProperties properties) {
            return new NettyRoutingFilter(httpClient, headersFilters, properties);
        }

2.global filter和gateway filter的联系
要说到他们的联系,我们知道不管是global filter还是gateway filter,他们都能够组成一个filter链来做拦截,而这个filter链是由List< GatewayFilter > 集合组成的,这看起来是GatewayFilter的组合,跟GlobalFilter无关。其实SCG通过适配器的方式,把GlobalFilter适配为GatewayFilter了,我们可以在FilteringWebHandler的构造器中看到这个装换:

    private final List<GatewayFilter> globalFilters;
    
    // 这里会把所有的global filter注入进来
    public FilteringWebHandler(List<GlobalFilter> globalFilters) {
        this.globalFilters = loadFilters(globalFilters);
    }

    private static List<GatewayFilter> loadFilters(List<GlobalFilter> filters) {
        return filters.stream().map(filter -> {
            // 遍历所有的global filter,适配成gateway filter
            GatewayFilterAdapter gatewayFilter = new GatewayFilterAdapter(filter);
            if (filter instanceof Ordered) {
                int order = ((Ordered) filter).getOrder();
                // OrderedGatewayFilter的定义与GatewayFilterAdapter类似,就是增加了order属性用来做优先级的排序
                return new OrderedGatewayFilter(gatewayFilter, order);
            }
            return gatewayFilter;
        }).collect(Collectors.toList());
    }

我们可以看看GatewayFilterAdapter的定义:

private static class GatewayFilterAdapter implements GatewayFilter {
        private final GlobalFilter delegate;
        GatewayFilterAdapter(GlobalFilter delegate) {
            // GlobalFilter是执行者
            this.delegate = delegate;
        }

        @Override
        public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
            // 通过委派的方式驱动GlobalFilter工作
            return this.delegate.filter(exchange, chain);
        }
    }

3.小结

简单的来说GlobalFilter是全局的filter所有都要走,而GatewayFilter必须要有固定的工厂去调用。至于作者为什么会定义两种filter,我认为是单一职责的原因,可以让开发者一看就知道哪个filter是有全局作用,哪个filter只针对某个route的请求起作用,职责分明。当要添加某种类型的filter时,只要实现其对应的接口即可。
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值