SpringCloudGateway 入门

news2024/11/19 15:14:13

目录

  • POM 依赖
  • 一、内容
    • 网关的作用
    • Spring-Cloud-Gateway的核心概念
  • 二、基于Ribbon的负载均衡
  • 三、核心概念详细
    • 3.1 断言 Predicate
    • 3.2 过滤器
      • 3.2.1 内置过滤器
      • 3.2.2 自定义过滤器
        • 构造器(原理)
        • 资源结构
          • Route / Predicate 的构造器
          • 构造器的增强器
          • 整体协同关系 / 部分源代码分析
      • 扩展自己的过滤器(实战)

POM 依赖

我的Java版本是17
父模块

<properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <alibaba-cloud.version>2022.0.0.0</alibaba-cloud.version>
    <spring-cloud.version>2022.0.4</spring-cloud.version>
    <spring-boot.version>3.0.9</spring-boot.version>
    <circuit-breaker.version>3.0.3</circuit-breaker.version>
</properties>

<dependencyManagement>
    <dependencies>
        <dependency>
        <groupId>com.alibaba.cloud</groupId>
        <artifactId>spring-cloud-alibaba-dependencies</artifactId>
        <version>${alibaba-cloud.version}</version>
        <type>pom</type>
        <scope>import</scope>
        </dependency>

        <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-dependencies</artifactId>
        <version>${spring-boot.version}</version>
        <type>pom</type>
        <scope>import</scope>
        </dependency>

        <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-dependencies</artifactId>
        <version>${spring-cloud.version}</version>
        <type>pom</type>
        <scope>import</scope>
        </dependency>
    </dependencies>
</dependencyManagement>

子模块

 <dependencies>
    <!-- 网关 -->
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-gateway</artifactId>
    </dependency>

    <!-- 注册中心配置 -->
    <dependency>
        <groupId>com.alibaba.cloud</groupId>
        <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
    </dependency>

    <!-- 注册中心客户端 -->
    <dependency>
        <groupId>com.alibaba.cloud</groupId>
        <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
    </dependency>
</dependencies>

一、内容

网关的作用

  • 假设:如果没有网关,客户端如何调用微服务,数十个微服务。
    如果真有数十个微服务,那客户端要记录数十个微服务的地址,然后分别去使用,这样会存在很多问题,比如:
    1. 重复造轮子
      每个微服务单独为战,各自实现自己的鉴权、限流、跨域等,把轮子重造。
    2. 调用低效
      如果业务量比较简单的话,暂时还不会有什么问题,随着业务越来越复杂,一个复杂的页面可能会涉及到数百个微服务协同工作,如果每个微服务都分配一个域名的话,一方面客户端代码会很难维护,涉及到数百个域名,另一方面是连接数的瓶颈,想象一下你打开一个APP,通过抓包发现涉及到了数百个远程调用,这在移动端下会显得非常低效。
    3. 重构复杂
      后期如果需要对微服务进行重构的话,也会变的非常麻烦,需要客户端配合你一起进行改造,比如商品服务,随着业务变的越来越复杂,后期需要进行拆分成多个微服务,这个时候对外提供的服务也需要拆分成多个,同时需要客户端配合你进行改造,非常蛋疼。

上面的问题,如果引入了网关,由网关统一管理,客户端只需要和网关对接,网关就负责授权、限流、路由转发等。

Spring-Cloud-Gateway的核心概念

  • Route:网关的基本构建块,它由一个ID、一个目标URI、一组断言和一组过滤器定义。如果聚合断言为真,则匹配路由。
  • Predicate:支持对请求头、请求方法、请求参数、Host、时间(设置过期时间/生效时间)等等进行匹配,匹配成功之后会执行Filter。
  • Filter:有修改url,修改body,添加请求头/请求参数,限流,断路器,权限认证等功能。

二、基于Ribbon的负载均衡

在网关模块添加这个依赖就行了,注意我这个是子模块,我的父模块管理了SpringCloud的依赖,所以不需要添加版本

<!-- 负载均衡器,用作服务发现,支持路由的 lb://servername -->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-loadbalancer</artifactId>
</dependency>

然后写路由就可以使用lb://servername了。

@Bean
public RouteLocator usrRout(RouteLocatorBuilder builder) {
    ZonedDateTime zonedDateTime = LocalDateTime.now()
            .plusSeconds(20)
            .atZone(ZoneId.systemDefault());

    return builder.routes()
            .route("user", r -> r.path("/xxxx").uri("lb://oraone"))
            .build();
}

三、核心概念详细

3.1 断言 Predicate

  1. 关于时间的
    before:在指定时间之后路由永远失效。
    after:在指定时间之后路由才生效。
    between:在指定时间之间生效,其余时间失效。
@Configure
public class Config {
	public RouteLocator config(RouteLocatorBuilder build) {
		ZoneDateTime before = ZoneDateTime.now()
			.plusSeconds(36)
			.atZone(ZoneId.systemDefault());
		ZoneDateTime after = ZoneDateTime.now()
			.plusSeconds(900)
			.atZone(ZoneId.systemDefault());
		return build.routes()
		// 在before之后永久路由失效
		.route("before", r -> r.before(before).uri("lb://servername"))
		// 在after之后路由生效。
		.route("after", r -> r.after(after).uri("lb://servername"))
		// 在[before, after]之内路由生效。
		.between("between", r -> r.between(before, after).uri("lb://servername"))
		.build();
	}
}
  1. 关于HTTP的
    Header:限制请求头和请求头的值
    Host:限制主机
    Method:限制请求方法
    Query:限制请求参数
@Configure
public class RouteConfig {
	@Bean
	public RouteLocator route(RouteLocatorBuilder build) {
		return build.routes()
			// 请求必须有name,和token这两个请求头,token必须是32位指定字符。
			// 请求方法必须是get,只能本地请求。
			// 参数必须携带id
			.route("route", r -> e.header("name")
				and().header("token", "[a-zA-Z0-9]{32, 32}")
				and().method(HttpMethod.GET)
				and().Host("localhost")
				and().query("id")
				.uri("lb://servername"))
			.build();
	}
}
  1. 关于路径的
    Path:限制访问的路径及协议 域名 端口之后的url
@Configure
public class RouteConfig {
	@Bean
	public RouteLocator route(RouteLocatorBuilder build) {
		return build.routes()
			.route("path", r -> r.path("/gateway/user/**").uri("lb://servername"))
			.build();
	}
}
  1. IP地址和网段
    RemoteAdd:两个参数,第一个是IP地址,第二个是网段。

  2. 权重
    Weight:设置接口的权重,权重越高,那么转发到该接口的请求数就越多。有两个参数,第一个个是GroupName,第二个是一个Weight数值。

@Configure
public class RouteConfig {
	@Bean
	public RouteLocator route(RouteLocatorBuilder build) {
		String path = "/gateway/user/**";
		String groupName = "user";
		return build.routes()
			.route("weight1", r -> r.path(path)
				and().weight(groupName, 80).uri("lb://servername1"))
			.route("weight2", r -> r.path(path)
				.and().weight(groupName, 20).uri("lb://servername2")))
			.build();
	}
}

3.2 过滤器

3.2.1 内置过滤器

内置的过滤器大体分为请求头、响应头、跳转、参数处理、响应状态、熔断、限速器。

  • 关于HTTP的过滤器
    AddRequestHeader:
    AddRequestParams:
    AddResponseHeader:
  • 关于路径的过滤器
    RewritePath:支持正则匹配替换
    StripPrefix:可以把前缀路径干掉
@Configure
public class RouteConfig {
	@Bean
	public RouteLocator route(RouteLocatorBuilder build) {
		return build.routes()
			.route("route", r -> r.path("/gateway/usr/local/**")
				.filters(f -> f.rewritePath("/gateway/usr/($<segment>.*)", "/user/$\\{segment}"))
				.uri("lb://servername"))
			// 最后访问路径是 lb://servername/order/**
			.route("route1", r -> r.path("/gateway/order/**")
				.filters(f -> f.stripPrefix(1)).uri("lb://servername"))
			.build();
	}
}

3.2.2 自定义过滤器

构造器(原理)
资源结构

gateway最终需要的资源是RouteLocator,RouteLocator里面包含了路由、断言、过滤器三个种资源,它们的依赖关系如下:

请添加图片描述

Route / Predicate 的构造器

Buildalbe就一个方法build,AbstractBuilder负责构建Route,源码如下:
五个属性,都是路由的基本组成,还看到了GatewayFilter 局部过滤器。

public abstract static class AbstractBuilder<B extends AbstractBuilder<B>> implements Buildable<Route> {
	protected String id;
	protected URI uri;
	protected int order = 0;
	protected List<GatewayFilter> gatewayFilters = new ArrayList<>();
	protected Map<String, Object> metadata = new HashMap<>();
    省略....
}

最后是Buidler和AsyncBuidler,它两负责构建断言,只是构建的类型不同,至于作用是什么目前还不清楚,它们两分别构建Predicate和AsyncPredicate。
请添加图片描述

构造器的增强器

SpringCloudGateway提供了很多GatewayFilter过滤器,UriSpec就是这些过滤器的配置,UriSpec里面包含了Route.Builder / RouteLocatorBuilder.Builder,配置的断言和过滤器最终都会流入Route.Builder。
请添加图片描述增强器和构造器的协作关系,Route.AsyncBuilder和RouteLocatorBuidler.Builder都组合在UriSpec中。
请添加图片描述

整体协同关系 / 部分源代码分析

请添加图片描述

public static class Builder {

	private List<Buildable<Route>> routes = new ArrayList<>();
	private ConfigurableApplicationContext context;
	public Builder(ConfigurableApplicationContext context) {
		this.context = context;
	}
   
   // 创建一个Buildable,放到routes中,ID是自命名。
   // apply会生成一个PredicateSpec,提供给用户选择过滤器,最后通过UriSpec的uri方法,返回一个Buildable对象。
	public Builder route(String id, Function<PredicateSpec, Buildable<Route>> fn) {
		Buildable<Route> routeBuilder = fn.apply(new RouteSpec(this).id(id));
		add(routeBuilder);
		return this;
	}
    
   // 创建一个Buildable,放到routes中,ID随机生成。
   // apply会生成一个PredicateSpec,提供给用户选择过滤器,最后通过UriSpec的uri方法,返回一个Buildable对象。
	public Builder route(Function<PredicateSpec, Buildable<Route>> fn) {
		Buildable<Route> routeBuilder = fn.apply(new RouteSpec(this).randomId());
		add(routeBuilder);
		return this;
	}
   
   // 执行所有的Buildable,构造RouteLocator。
	public RouteLocator build() {
		return () -> Flux.fromIterable(this.routes).map(routeBuilder -> routeBuilder.build());
	}

	ConfigurableApplicationContext getContext() {
		return context;
	}

	void add(Buildable<Route> route) {
		routes.add(route);
	}
}
public class UriSpec {

	final Route.AsyncBuilder routeBuilder;

	final RouteLocatorBuilder.Builder builder;

	UriSpec(Route.AsyncBuilder routeBuilder, RouteLocatorBuilder.Builder builder) {
		this.routeBuilder = routeBuilder;
		this.builder = builder;
	}

	public UriSpec customize(Consumer<Route.AsyncBuilder> routeConsumer) {
		routeConsumer.accept(this.routeBuilder);
		return this;
	}

	public UriSpec replaceMetadata(Map<String, Object> metadata) {
		this.routeBuilder.replaceMetadata(metadata);
		return this;
	}

	public UriSpec metadata(Map<String, Object> metadata) {
		this.routeBuilder.metadata(metadata);
		return this;
	}

	public UriSpec metadata(String key, Object value) {
		this.routeBuilder.metadata(key, value);
		return this;
	}

	// 设置uri,这里会抛出一个Buildalbe,对应构造器的写法:build.routes().route(r -> r.uri()).build();
	public Buildable<Route> uri(String uri) {
		return this.routeBuilder.uri(uri);
	}

	/**
	 * Set the URI for the route.
	 * @param uri the URI for the route.
	 * @return a {@link Route.AsyncBuilder}S
	 */
	public Buildable<Route> uri(URI uri) {
		return this.routeBuilder.uri(uri);
	}

	<T> T getBean(Class<T> type) {
		return this.builder.getContext().getBean(type);
	}
}

扩展自己的过滤器(实战)

GatewayFilterSpec有filter方法,可以直接添加自己的过滤器。

@Configure
public class Config{
    @Bean
    public RouteLocator usrRout(RouteLocatorBuilder builder) {
        ZonedDateTime zonedDateTime = LocalDateTime.now()
            .plusSeconds(20)
            .atZone(ZoneId.systemDefault());

    return builder.routes()
            .route("user", r -> r.before(zonedDateTime).uri("lb://oraone"))
            .route("user1", predicate -> predicate
                    .path("/strip/prefix/**")
                    .filters(filter -> filter.stripPrefix(2)
                            // 自定义GatewayFilter
                            .filter(new MyGatewayFilter()))
                    .uri("lb://oraone"))
            .build();
    }
    
    class MyGatewayFilter implements GatewayFilter {
        @Override
        public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
            // 什么都不做。
            return chain.filter(exchange);
        }
    }
}

还可以继承GatewayFilterFactory,实现它的apply方法。或者继承AbstractGatewayFilterFactory,实现apply方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1141733.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

人大金仓(Kingbase)部署

点击上方蓝字关注我 1. 介质下载 下载地址&#xff1a;https://www.kingbase.com.cn/rjcxxz/index.htm 选择安装包及授权文件&#xff1a;根据对应的操作系统类型选择安装包 2. 部署环境配置 2.1 部署环境&#xff1a; 8C 16G KylinV10SP3系统 2.2 修改操作系统内核参数 sy…

什么是pmp证书,pmp证书有什么用,pmp项目管理证书的认证考试时间是什么时候啊?

PMP是项目管理证书&#xff0c;目标是项目经理。 英文全称是Project Management Professional&#xff0c;中文全称叫做项目管理专业人士资格认证。 它是由美国项目管理协会&#xff08;PMI&#xff09;在全球范围内推出的针对项目经理的资格认证体系&#xff0c;严格评估项目…

贪心算法总结(未完结)

贪心的定义&#xff08;摘自百度百科&#xff09; 贪心算法&#xff08;greedy algorithm&#xff0c;又称贪婪算法&#xff09;是指&#xff0c;在对问题求解时&#xff0c;总是做出在当前看来是最好的选择。也就是说&#xff0c;不从整体最优上加以考虑&#xff0c;算法得到的…

LeetCode刷题:26. 删除有序数组中的重复项

文章目录 写在前面⭐️26. 删除有序数组中的重复项⭐️&#x1f510;题目描述&#x1f4a1;解题思路&#x1f511;代码 写在前面 本题的题解代码是用C语言编写的。 &#x1f4d2;博客主页&#xff1a;2023Fighting的博客主页 &#x1f389;欢迎关注&#x1f50e;点赞&#x1f…

Springboot 使用JavaMailSender发送邮件 + Excel附件

目录 1.生成Excel表格 1.依赖设置 2.代码&#xff1a; 2.邮件发送 1.邮件发送功能实现-带附件 2.踩过的坑 1.附件名中文乱码问题 3.参考文章&#xff1a; 需求描述&#xff1a;项目审批完毕后&#xff0c;需要发送邮件通知相关人员&#xff0c;并且要附带数据库表生成的…

[ubuntu系统下的文本编辑器nano,vim,gedit,文件使用,以及版本更新问题]

文本编辑器概要 在Ubuntu系统下&#xff0c;有许多文本编辑器可供选择&#xff0c;每个编辑器都有其独特的特性和用途。以下是一些常见的文本编辑器&#xff1a; Gedit&#xff1a; 这是Ubuntu默认的文本编辑器&#xff0c;它简单易用&#xff0c;适合基本的文本编辑任务。 安…

Java采集传感器数据,亲测有效!

背景 先说背景&#xff0c; 最近公司项目需要用到传感器&#xff0c;采集设备温湿度&#xff0c;倾斜角&#xff0c;电流…&#xff0c;公司采购采购了一个温湿度传感器给我们开发测试使用&#xff0c;如下图&#xff1a; 看着还挺精致有没有。 进入正题 有了这个温湿度传感器…

【Leetcode】【每日一题】【中等】1465. 切割后面积最大的蛋糕

力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能&#xff0c;轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/maximum-area-of-a-piece-of-cak…

Vue3+Element-Plus项目 el-table 拖拽排序实现,Vue3项目sortablejs的安装与使用

概述 技术栈&#xff1a; Vue3 Ts Vite Element-Plus 实现&#xff1a;实现 sortablejs 实现 el-tabel 的拖拽排序&#xff0c;可滚动排序&#xff0c;并实现拖拽排序的开启与关闭 文章目录 概述一、先看效果二、安装 sortablejs三、sortablejs 封装3.1 utilts 封装3.2 全局…

SpringBoot 定时任务:@EnableScheduling @Scheduled

Scheduled注解参数 cron参数 这个参数是最经常使用的参数&#xff0c;表示接收一个cron参数&#xff0c;cron它是一个表达式&#xff0c;最多接收7个参数&#xff0c;从左到右分别表示&#xff1a;秒 分 时 天 月 周 年&#xff1b;参数以空格隔开&#xff0c;其中年不是必须参…

【JAVA学习笔记】48 - 八大常用Wrapper类(包装类)

一、包装类 1.针对八种基本定义相应的引用类型一包装类 2.有了类的特点&#xff0c;就可以调用类中的方法。 黄色背景的表示父类是Number 二、包装类和基本数据的转换 演示包装类和基本数据类型的相互转换&#xff0c;这里以int和Integer演示。 1.jdk5前的手动装箱和拆箱方…

STM32 TIM(四)编码器接口

STM32 TIM&#xff08;四&#xff09;编码器接口 编码器接口简介 Encoder Interface 编码器接口 编码器接口可接收增量&#xff08;正交&#xff09;编码器的信号&#xff0c;根据编码器旋转产生的正交信号脉冲&#xff0c;自动控制CNT自增或自减&#xff0c;从而指示编码器的…

取Dataset子集(pytorch)

取Dataset子集--pytorch 1. why2. how3. example 1. why 我们在调试深度学习代码时&#xff0c;常常会遇到数据集太大&#xff0c;导致调试浪费时间的情况&#xff0c;这种情况下&#xff0c;将数据集中的一个子集拿出来用于调试代码&#xff0c;调试成功在用完整的数据集运行…

elementUI el-table实现鼠标悬浮某一行,在鼠标右侧展示提示信息

背景 el-table组件中&#xff0c;可以通过勾选某条数据来创建单据&#xff0c;但是有些数据没有权限使用&#xff0c;就需要禁用掉勾选的功能&#xff0c;然后当鼠标悬浮在这一行的时候&#xff0c;展示类似于toolTip的提示框。 除了当鼠标悬浮在某一行&#xff0c;展示类似于…

离散数学速成视频推荐(讲的不错)

【拯救者】离散数学速成(期末考研专升本) 适合人群 期末 考研 复试 转升本 月考都可以用 课程大纲 适用课本 适用于所有离散数学课本 按课本章节来&#xff0c; 抽取重点&#xff0c;翻译为人话 学习步骤&#xff1a; 每一章&#xff0c;都会【讲会考的基础&#x1f4bb;】&a…

跨平台Markdown编辑软件Typora mac中文版功能介绍

Typora mac是一款跨平台的Markdown编辑器&#xff0c;支持Windows、MacOS和Linux操作系统。它具有实时预览功能&#xff0c;能够自动将Markdown文本转换为漂亮的排版效果&#xff0c;让用户专注于写作内容而不必关心格式调整。Typora Mac版除了支持常见的Markdown语法外&#x…

通俗易懂的理解 解耦 概念

解耦&#xff08;Decoupling&#xff09;是计算机科学和软件工程中的一个概念&#xff0c;指的是降低系统中不同部分之间的依赖性&#xff0c;使系统的各个组件能够相对独立地进行开发、维护和演化。解耦的主要目标是减少组件之间的紧密耦合&#xff0c;以提高系统的灵活性、可…

Tomcat的动静分离

一、动态负载均衡 3、台虚拟机模拟&#xff1a; 代理服务器&#xff1a;51 tomcat动态页面&#xff1a;53,54 关闭防火墙和安全机制 配置代理服务器&#xff0c;由于做的是七层代理&#xff0c;所以要在http模块配置 配置前端页面 <!DOCTYPE html> <html> <…

AutoEncoding与AutoRegressive:区别,联系和应用

关于AutoEncoding&#xff08;AE&#xff09;和AutoRegressive&#xff08;AR&#xff09; 前几天看了Ilya在Simons上做的关于Generative Model的演讲&#xff0c;介绍了OpenAI现在做的一些AutoRegressive的工作&#xff0c;昨天又看到LeCun宣称Auto-Regressive LLMs are doom…

ES6新特性:变量的解构赋值

文章目录 1 数组的解构赋值1.1 基本用法1.2 交换变量的值1.3 注意事项 2 对象的解构赋值2.1 基本用法2.2 属性重命名2.3 注意事项 ES6允许按照一定模式&#xff0c;从数组和对象中提取值&#xff0c;对变量进行赋值&#xff0c;这被称为解构&#xff08;Destructuring&#xff…