1.什么是网关
Gateway(网关)是一种计算机网络设备,它可以连接两个或多个不同的网络,将它们之间的数据进行转换和传递,实现不同网络之间的互联互通。在物联网中,Gateway通常指的是边缘计算设备,它可以连接物联网设备和云端服务,实现数据的采集、处理和传输。Gateway的作用类似于一个桥梁,将物联网设备和云端服务连接起来,实现数据的无缝传输和处理。Gateway通常具有较强的计算和存储能力,可以在设备端进行数据处理和分析,减少数据传输的延迟和带宽占用,提高数据的安全性和可靠性。
1.2使用网关的优点
(1)简化客户端的工作。网关将微服务封装起来后,客户端只需同网关交互,而不必调用各个不同服务;
(2)降低函数间的耦合度。 一旦服务接口修改,只需修改网关的路由策略,不必修改每个调用该函数的客户端,从而减少了程序间的耦合性
(3)解放开发人员把精力专注于业务逻辑的实现。由网关统一实现服务路由(灰度与ABTest)、负载均衡、访问控制、流控熔断降级等非业务相关功能,而不需要每个服务 API 实现时都去考虑,可以通过网关的过滤器,对url进行处理,如在原有的url上添加元素,或者获取形参信息,或拦截请求。
1.3 请求流程
2.快速搭建网关
创建spring boot 项目 ,导入Gateway依赖和Eureka客户端依赖,注意不用加入MVC依赖,它与Gateway冲突,因为Gateway内置了Mvc。
2.1导入 依赖
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
2.1 编写yml配置文件
#改模块端口号
server:
port: 8089
spring:
application:
name: gateway #服务名称
cloud:
gateway:
routes:
- id: order #名称随意
uri: lb://user # lb;负载均衡 + 服务名称 或者使用http
predicates:
- Path=/user/** # url路径
- id: user
uri: lb://order
predicates:
- Path=/order/**
# eureka 服务地址
eureka:
client:
service-url:
defaultZone: http://127.0.0.1:8082/eureka/
2.2 配置 消费者模块 和 提供者模块与 EureKa 服务端,之后先启动Eureka 之后启动其他模块。
加入EureKa成功
代码在往期的博客中
3 Spring Cloud Gateway 配置项的说明:
在介绍 Spring Cloud Gateway 的配置项之前,我们先了解几个 Spring Cloud Gateway 的核心术语:
断言(Predicate):参照 Java8 的新特性Predicate,允许开发人员匹配 HTTP 请求中的任何内容,比如请求头或请求参数,最后根据匹配结果返回一个布尔值。
路由(route):由ID、目标URI、断言集合和过滤器集合组成。如果聚合断言结果为真,则转发到该路由。
过滤器(filter):可以在返回请求之前或之后修改请求和响应的内容。
其中路由包括:
Route 主要由 路由id、目标uri、断言集合和过滤器集合组成,那我们简单看看这些属性到底有什么作用。
(1)id:路由标识,要求唯一,名称任意(默认值 uuid,一般不用,需要自定义)
(2)uri:请求最终被转发到的目标地址
(3)order: 路由优先级,数字越小,优先级越高
(4)predicates:断言数组,即判断条件,如果返回值是boolean,则转发请求到 uri 属性指定的服务中。
(5)filters:过滤器数组,在请求传递过程中,对请求做一些修改
4.配置过滤器 Filters
局部过滤器,配置在路由列表用户下
cloud:
gateway:
routes:
- id: order #服务名称随意
uri: lb://user # lb;负载均衡 + 服务名称 或者http
predicates:
- Path=/user/** # url路径
filters:
- AddRequestHeader=tru, kk # tru为url上参数名称 kk 为 给参数赋值的属性
全局过滤器,注意它与routes是同一级别下
cloud:
gateway:
routes:
- id: order #服务名称随意
uri: lb://user # lb;负载均衡 + 服务名称 或者http
predicates:
- Path=/user/** # url路径
default-filters:
- AddRequestHeader=tru,it
GlobalFilter,需要实现GlobalFilter接口,实现Filter方法,这段代码是先获取请求参数,之后获取形参 参数判断,成功交给下一个过滤器,失败返回前端状态码 401
/*
作用
拦截 请求,参数判断是否满足
exchange: 可以获取 请求方式 ,josn 信息等
chain :把请求委托给下一个过滤器,放行
*/
// 这是过滤器优先级
@Order(1)
@Controller
public class gate implements GlobalFilter {
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
//获取请求参数
ServerHttpRequest request = exchange.getRequest();
ServerHttpResponse response = exchange.getResponse();
MultiValueMap<String, String> queryParams = request.getQueryParams();
MultiValueMap<String, ResponseCookie> cookies = response.getCookies();
System.out.println(cookies.size());
// 获取 请求中 orderId 参数
String orderId = queryParams.getFirst("orderId");
// 判断是否符合要求
if(orderId.equals("103")){
//符合放行
Mono<Void> filter = chain.filter(exchange);
return filter;
}
//不符合 ,拦截,建议设置状态码,否则拦截请求什么都不返回
// 设置状态码
exchange.getResponse().setStatusCode(HttpStatus.UNAUTHORIZED);
return exchange.getResponse().setComplete();
}
}
效果演示
GlobalFilter,
这个1003是数据库中没有的数据,也就是说在进入服务前拦截下了
局部与全局过滤器效果
三种过滤器优先级
order一样时 default-filters>filters>GlobalFilter