[SpringCloud]Gateway入门


Gateway

Gateway和Zuul的理念差别

SpringCloud中集成的Zuul版本,采用的是Tomcat容器,使用的是传统的Servlet IO处理模型。即:

container启动的时候构造Servlet对象并调用Servlet.init()方法进行初始化。          container运行时接收请求,并为每一个请求分配一个线程(一般从线程池获取空闲线程),然后调用service()。          container关闭时调用Servlet.destroy()销毁Servlet。

这样的模式的缺点是,并发量高的情况下,大量的Servlet产生,从而绑定了大量的线程,而线程的资源实际上是很昂贵的,上下文的切换和内存的消耗会导致请求的处理时间延长。

而Gateway的底层使用的是netty和webflux,webflux是一个典型的非阻塞异步框架,核心是基于Reactor的相关API实现的,相对于传统的Web框架来说,可以运行在诸如netty,undertow及支持Servlet3.1的容器上。

Gateway的三大核心概念

路由Route

Route是构建网关的基本模块,由ID,目标URI,一系列的断言和过滤器组成,如果断言为true则匹配这个路由。

断言Predicate

参考的是java8中的java.util.function.Prediacate,开发人员可以匹配HTTP请求中的所有内容,如果请求和断言相匹配则进行路由。

过滤器Filter

指的是Spring框架中的GatewayFilter的实例,使用过滤器,可以在请求路由前或者之后对请求进行修改。

Gateway工作流程

客户端向SpringCloudGateway发出请求,然后在Gateway Handler Mapping中找到与请求相匹配的路由,将其发送到Gateway Web Handler,Handler再通过制定的过滤器来将请求发送到我们实际的服务执行业务逻辑,然后返回。简单的老说,它相当于一个门卫,挡在我们的服务之前,在请求到达我们的服务的时候能够做一些前置的处理,对于客户来说背后的服务实际上是屏蔽的,与我们交互的是网关。

GateWay的基本使用

插一句话,最近刚参加工作,深感比在学校的时候要操心的地方要多得多,留给自己学习的时间也远远没有学校多,只能趁周六周末来充电,实际上这个博客在上个月月底就已经起草,断断续续没写几个字,而且Gateway的基本的东西也没学完,所以一直拖到今天,今天也不打算把Gateway学个底朝天,只打算总结一下之前学的基本的东西,更多东西还是要去实际项目中用再能总结出来,就先这样吧。
在写Gateway的模块之前,事实上肯定至少有一个服务和一个注册中心的,这里也不讲Eureka和那个无关紧要的服务了。

配置Gateway的依赖

    

        
        
            junit
            junit
            4.12
            test
        

        
            cn.izzer
            cloud-api-common
            1.0
        
        
        
            org.springframework.cloud
            spring-cloud-starter-gateway
        
        
        
            org.springframework.cloud
            spring-cloud-starter-netflix-eureka-client
        
        
            org.projectlombok
            lombok
            true
        
        
            org.springframework.boot
            spring-boot-starter-test
            test
        
    

yml配置

这里主要是通过配置文件来配置一些映射,还有定义注册中心地址什么的,之后还有通过代码来配置全局的过滤器等等。

server:
  port: 9527
spring:
  application:
    name: cloud-gateway
  cloud:
    gateway:
      discovery:
        locator:
          enabled: true #开启动态路由
      routes:
        - id: payment_routh1
          uri: http://localhost:8001
          predicates:
            - Path=/payment/get/** #配置路径映射
            #- After=2020-08-14T22:44:44.829+08:00[Asia/Shanghai] 主要检测时间
            #- Cookie=username,zzyy 检测cookie,可以用curl进行模拟
            #- Header=X-Request-Id,\d+ #请求头要有X-Request-Id属性并且值为整数的正则表达式
        - id: payment_routh2
          uri: http://localhost:8001
          predicates:
            - Path=/payment
eureka:
  instance:
    hostname: cloud-gateway-service
  client:
      register-with-eureka: true
      fetch-registry: true
      service-url:
        defaultZone: http://localhost:7001/eureka

上面是配置了路径映射,发起请求的时间拦截,cookie等等,时间的设置需要调用ZoneDateTime来得到。如下:

    @Test
    public void timeZoneTest(){
        ZonedDateTime time = ZonedDateTime.now();
        System.out.println(time);
    }

这里演示一下Cookie和Header的测试,因为这里用到curl,我很少用这个东西,都是用postman,今天来涨涨知识。
avatar
然后是测试Header(乱码了......)
avatar
官网上还有许多相类似的配置,就不一一展示了,也没多大意义。接下来是用代码实现路由配置。主要用到的是RouteLocatorBuilder.Builder。用@Configuration加上注解即可。

@Configuration
public class GateWayConfig {
    @Bean
    public RouteLocator customRouteLocator(RouteLocatorBuilder routeLocatorBuilder){
        //这里是将网关服务的/guonei映射到百度新闻的国内新闻版块。
        RouteLocatorBuilder.Builder builder = routeLocatorBuilder.routes();
        builder.route("path_route_atguigu",r->r.path("/guonei").uri("http://news.baidu.com/guonei"));
        return builder.build();
    }
}

测试效果:
avatar

Filter组件

物如其名,做过滤用的,这里演示一下全局判断接口传入的字段(在实际工作这个不要乱用)。主要是要实现GlobalFilter和Ordered接口,然后在里面加上自己的业务逻辑。我这里简单判断一下uname字段是否传入。

@Component
@Slf4j
public class GlobalGatewayFilter implements GlobalFilter,Ordered {
    @Override
    public Mono filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        log.info("------- come into global filter------");
        String uname = exchange.getRequest().getQueryParams().getFirst("uname");
        if (uname == null){
            log.info("---------uname is null ----- ");
            exchange.getResponse().setStatusCode(HttpStatus.NOT_ACCEPTABLE);
            return exchange.getResponse().setComplete();
        }
        return chain.filter(exchange);
    }
    @Override
    public int getOrder() {
        return 0;
    }
}

传入uname的情况:
avatar
没有传入的情况:
avatar

参考

B站上的SpringCloud,适合跟着做个demo入门