什么是网关

image.png
image.png
如上图所示
API网关是一个搭建在客户端和微服务之间的服务,我们可以在API网关中处理一些非业务功能的逻辑,比如权限验证、监控、缓存、请求路由等等

客户端在发起请求的时候,会将请求发送到微服务网关中,然后再由网关将请求转发到微服务实例。 因此,客户端只用知道网关的地址即可,而不需要维护大量的服务地址,简化了客户端的配置。 在网关中,我们进行一些额外的操作,比如鉴权、流控、过滤等等

SpringCloud Gateway

SpringCloud Gateway是基于WebFlux框架实现的,而WebFlux框架底层则是使用了高性能的Reactor模式的通信框架Netty

WebFlux

与传统的SpringMVC等基于Servlet API和Servlet容器基础之上运行的,WebFlux是一个典型的非阻塞异步的框架

工作流程

  1. 路由(Route):路由是构建网关的基本模块,它由ID,目标URI,一系列的断言和过滤器组成,如断言为true则匹配该路由;
  2. 断言(Predicate):参考的是Java8的java.util.function.Predicate,开发人员可以匹配HTTP请求中的所有内容(例如请求头或请求参数),如果请求与断言相匹配则进行路由;
  3. 过滤(Filter):指的是Spring框架中的GatewayFilter的实例,使用过滤器,可以在请求被路由前或者之后对请求进行修改。

    先说断言,这个东西其实可以看成一个布尔表达式。如果为真就啥都不干,放你过去。如果为假,即断言不成立,就不让你过去 接着说下路由,首先一个请求进来之后,网关会根据请求的URL转发到对应的路由上。而每个路由上又会有对应的断言,上文说了,断言就是个布尔表达式。它会判断请求的路径是否符合要求。如果符合要求则让你进路由,否则,哪凉快去哪里。

测试项目搭建

新建模块

新建Module - cloud-gateway-gateway9527

  1. <dependencies>
  2. <!--gateway-->
  3. <dependency>
  4. <groupId>org.springframework.cloud</groupId>
  5. <artifactId>spring-cloud-starter-gateway</artifactId>
  6. </dependency>
  7. <!--eureka-client-->
  8. <dependency>
  9. <groupId>org.springframework.cloud</groupId>
  10. <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
  11. </dependency>
  12. <!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
  13. <dependency>
  14. <groupId>com.lun.springcloud</groupId>
  15. <artifactId>cloud-api-commons</artifactId>
  16. <version>${project.version}</version>
  17. </dependency>
  18. <!--一般基础配置类-->
  19. <dependency>
  20. <groupId>org.springframework.boot</groupId>
  21. <artifactId>spring-boot-devtools</artifactId>
  22. <scope>runtime</scope>
  23. <optional>true</optional>
  24. </dependency>
  25. <dependency>
  26. <groupId>org.projectlombok</groupId>
  27. <artifactId>lombok</artifactId>
  28. <optional>true</optional>
  29. </dependency>
  30. <dependency>
  31. <groupId>org.springframework.boot</groupId>
  32. <artifactId>spring-boot-starter-test</artifactId>
  33. <scope>test</scope>
  34. </dependency>
  35. </dependencies>
  1. server:
  2. port: 9527
  3. spring:
  4. application:
  5. name: cloud-gateway
  6. eureka:
  7. instance:
  8. hostname: cloud-gateway-service
  9. client: #服务提供者provider注册进eureka服务列表内
  10. service-url:
  11. register-with-eureka: true
  12. fetch-registry: true
  13. defaultZone: http://eureka7001.com:7001/eureka

配置网关

  1. server:
  2. port: 9527
  3. spring:
  4. application:
  5. name: cloud-gateway
  6. #############################新增网关配置###########################
  7. cloud:
  8. gateway:
  9. routes:
  10. - id: payment_routh #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
  11. uri: http://localhost:8001 #匹配后提供服务的路由地址
  12. #uri: lb://cloud-payment-service #匹配后提供服务的路由地址
  13. predicates:
  14. - Path=/payment/get/** # 断言,路径相匹配的进行路由
  15. ####################################################################
  16. eureka:
  17. instance:
  18. hostname: cloud-gateway-service
  19. client: #服务提供者provider注册进eureka服务列表内
  20. service-url:
  21. register-with-eureka: true
  22. fetch-registry: true
  23. defaultZone: http://eureka7001.com:7001/eureka

分别访问http://localhost:8001/payment/get/one/31http://localhost:9527/payment/get/one/31
会发现两者都可以访问成功,并且返回相同的结果

配置网关的方式

通过yml配置文件的方式

通过yml配置文件进行配置的方式如上代码块所示

通过Bean注入配置类的方式

新建GatewayConfig的类
通过路由的方式,将http://news.baidu.com/guonei映射到http://localhost:9527/guonei

  1. /**
  2. * Created By Intellij IDEA
  3. *
  4. * @author ssssheep
  5. * @package org.example.springcloud.config
  6. * @datetime 2022/8/11 星期四
  7. */
  8. @Configuration
  9. public class GatewayConfig {
  10. @Bean
  11. public RouteLocator customRouteLocator(RouteLocatorBuilder builder) {
  12. RouteLocatorBuilder.Builder routes = builder.routes();
  13. return routes.route("path_route_payment",
  14. r -> r.path("/guonei").uri("http://news.baidu.com/guonei"))
  15. .build();
  16. }
  17. }

image.png
可以发现,能够正常访问。

配置动态路由

在我们的yaml配置文件中,我们将请求的实际地址固定写死了,然而实际上我们提供服务的机器可能不止一台,这样的话,我们实际请求的地址应该是从注册中心处获取的,而不是固定写死的
image.png
因此,我们应这样来配置动态路由:
请求路径只写我们的服务名称,实际调用的是哪个微服务,由负载均衡的算法来决定
image.png
再次请求网关上的接口,可以发现8001和8002两个端口切换出现

未完待续

Predicate是什么

Filter