统一服务入口-Gateway(一)

发布于:2024-08-21 ⋅ 阅读:(131) ⋅ 点赞:(0)

目录

1.网关介绍

1.1含有问题

1.2什么是API网关

网关核心功能:

2.Spring Cloud Gateway

2.1什么是Spring Cloud Gateway

2.2快速上手

2.2.1创建网关项目

2.2.2引入网关依赖

2.2.3添加Gateway的路由配置

2.2.4测试

2.3Predicate

2.3.1Predicate的其他写法

2.3.2Predicate的其他方法

2.4Route Predicate Factories

2.4.1代码演示


1.网关介绍

1.1含有问题

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

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

比如企业管理
外部人员去公司办理业务,公司需要先核实对方的身份再去进行办理. 最开始只有一个员工,这个员工核实之后直接办理即可.(单体架构)

随着公司的发展,划分了多个部门,每个部门负责的事情不同,每个部门都需要先核实对方的身份再进行办理.(微服务架构)
 这个流程存在一些问题:
 1.办事效率低
 2.增加了员工的工作流程
我们对此进行改进,设立前台,统一由前台来进行身份的校验,前台身份校验通过后,其他部门就设置信任,直接办理.

1.2什么是API网关

API网关(简称网关)也是一个服务,通常是后端服务的唯一入口.

它的定义类似设计模式中的Facade模式 (门面模式,也称外观模式).它就类似整个微服务架构的门面,所有的外部客户端访问,都需要经过它来进行调度和过滤.

网关核心功能:

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

动态路由:一切请求先经过网关,网关不处理业务,而是根据某种规则,把请求转发到某个微服务

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

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

2.Spring Cloud Gateway

2.1什么是Spring Cloud Gateway

Spring Cloud Gateway是Spring Cloud的一个全新的API网关项目,基于Spring+SpringBoot等技术 开发,目的是为了替换掉Zuul.旨在为微服务架构提供一种简单而有效的途径来转发请求,并为他们提供横切关注点,比如:安全性,监控/指标和弹性.

在性能方面,根据官方提供的测试报告,SpringCloud Gateway的RPS(每秒请求数)是Zuul的1.6倍.

2.2快速上手

2.2.1创建网关项目

2.2.2引入网关依赖

 <!--⽹关-->
 <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>

2.2.3添加Gateway的路由配置

创建application.yml文件,添加如下配置:

server:
  port: 10030
spring:
  application:
    name: gateway
  cloud:
    nacos:
      discovery:
        server-addr: 110.41.51.65:10020
    gateway:
      metrics:
        enabled: true
      routes:
        - id: order-service   #路由规则id, 随便起, 不重复即可
          uri: lb://product-service/ #目标服务地址
          predicates:   #路由条件
            - Path=/order/**,/feign/**,/product/**
            - After=2024-03-20T00:00:22.370856700+08:00[Asia/Shanghai]

配置字段说明:

  • id:自定义路由ID,保持唯一
  • uri:目标服务地址,支持普通URI及lb://应用注册服务名称.lb表示负载均衡,使用lb://方式表示从注册中心获取服务地址.
  • predicates:路由条件,根据匹配结果决定是否执行该请求路由,上述代码中,我们把符合Path规则的一切请求,都代理到uri参数指定的地址.

2.2.4测试

启动API网关服务

1.通过网关服务访问product-service

url符合yml文件中配置的/product/**规则,

路由转发到product-service:http://product-service/product/1001

访问时,观察网关日志,可以看到网关服务从Nacos时获取服务列表

2.3Predicate

Predicate是Java 8提供的一个函数式编程接口,它回收一个参数并返回一个布尔值,用于条件过滤,请求参数的校验

 @FunctionalInterface
 public interface Predicate<T> {
      boolean test(T t);
      //...
 }

代码演示:

1.定义一个Predicate

 class StringPredicate implements Predicate<String>{
     @Override
     public boolean test(String str) {
         return str.isEmpty();
     }
 }

2.使用这个Predicate

 public class PredictTest {
     public static void main(String[] args) {
         Predicate<String> predicate = new StringPredicate();
         System.out.println(predicate.test(""));//true
         System.out.println(predicate.test("666"));//false
     }
 }

2.3.1Predicate的其他写法

1.内置函数

 public class PredictTest {
     public static void main(String[] args) {
         Predicate<String> predicate = new Predicate<String>(){
             @Override
             public boolean test(String s) {
                 return s.isEmpty();
             }
         };
         System.out.println(predicate.test(""));
         System.out.println(predicate.test("666"));
     }
 }

2.lambda写法

public class PredictTest {
    public static void main(String[] args) {
        Predicate<String> predicate = s -> s.isEmpty();
        System.out.println(predicate.test(""));
        System.out.println(predicate.test("bite666"));
    }
}

Predicate<String>predicate =s ->s.isEmpty();

也可以写成 Predicate<String>isEmpty =String::isEmpty;

2.3.2Predicate的其他方法

  • isEqual(Object targetRef):比较两个对象是否相等,参数可以为Null
  • and(Predicate other):短路与操作,返回一个组成Predicate
  • or(Predicate other):短路或操作,返回一个组成Predicate
  • test(T t):传入一个Predicate参数,用来做判断
  • negate():返回表示此Predicate逻辑否定的Predicate

2.4Route Predicate Factories

Route Predicate Factories(路由断言工厂,也称为路由谓词工厂,此处谓词表示一个函数),在Spring Cloud Gateway中,Predicate提供了路由规则的匹配机制.

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

比如前面章节配置的Path=/product/**,就是通过Path属性来匹配URL前缀是/product的 请求.

这个规则是由org.springframework.cloud.gateway.handler.predicate.PathRoutePredicateFactory来实现的.

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

2.4.1代码演示

1.添加Predicate规则

在application.yml中添加如下规则

spring:
  application:
    name: gateway
  cloud:
    nacos:
      discovery:
        server-addr: 127.0.0.1:8848
        namespace: b460e95c-d4b9-42a3-810f-cdf0051ce008
    gateway:
      metrics:
        enabled: true
      routes:
        - id: order-service   #路由规则id, 随便起, 不重复即可
          uri: lb://product-service/ #目标服务地址
          predicates:   #路由条件
            - Path=/product/**
            - After=2025-01-01T00:00:00.000+08:00[Asia/Shanghai]

添加限制规则:请求时间为2025年1月1日后

2.测试

访问:http://127.0.0.1:10030/product/1001

3.修改时间为2024-01-01,再次访问

- After=2024-01-01T00:00:00.000+08:00[Asia/Shanghai]


网站公告

今日签到

点亮在社区的每一天
去签到