SpringCloud Alibaba
根据官方文档学习(仅供参考和交流使用)
参考博客:
学习前思考:
:smile: 卷起来/卷起来/...
😂 抖音
、游戏
、农药
:cry: 一日复一日
:earth_africa: 人生の意味は何ですか
:satisfied:卷斯拉/卷斯拉/
生态圈:
"瞻前顾后"
不同厂商结合其自身的中间件,提供自己的 Spring Cloud 套件,例如说:
Netflix 结合自己的
Eureka
(淘汰)、Ribbon
、Hystrix
(过时) 等开源中间件,实现了 spring-cloud-NetflixSpring Cloud Netflix 是目前国内使用最为流行的 Spring Cloud 套件。不过随着 Netflix 在开源的调整,逐步会慢慢没落。
- Kubernetes 结合自己的
apiserver
、configmap
等功能,实现了 spring-cloud-kubernetes Alibaba 结合自己的
Nacos
、Dubbo
、Sentinel
等开源中间件,实现了 spring-cloud-alibaba目测 Spring Cloud Alibaba 在国内会火?!
Spring Cloud Alibaba 套件
Spring Cloud Alibaba 套件,阿里开源组件、阿里云商业组件整合进 Spring Cloud 体系当中,
同时对 Spring Cloud Gateway、OpenFeign、Ribbon 等等进行集成。整体如下图所示:
主要功能如下:
- 服务注册与发现:适配 Spring Cloud 服务注册与发现标准,默认集成了 Ribbon 的支持。
- 分布式配置管理:支持分布式系统中的外部化配置,配置更改时自动刷新。
- 服务限流降级:默认支持
WebServlet
、WebFlux
,OpenFeign
、RestTemplate
、Spring Cloud Gateway
,Zuul
,Dubbo
和RocketMQ
限流降级功能的接入,可以在运行时通过控制台实时修改限流降级规则,还支持查看限流降级Metrics
监控。 - 消息驱动能力:基于
Spring Cloud Stream
为微服务应用构建消息驱动能力。 - 分布式事务:使用
@GlobalTransactional
注解, 高效并且对业务零侵入地解决分布式事务问题。
商业化独有功能如下:
分布式任务调度:提供秒级、精准、高可靠、高可用的定时(基于
Cron
表达式)任务调度服务。同时提供分布式的任务执行模型,如网格任务。网格任务支持海量子任务均匀分配到所有 Worker(schedulerx-client
)上执行。曾经,阿里
SchedulerX
也是一个开源项目,现在已经木有消息了...嘿嘿,希望未来能够重启。- 阿里云对象存储:阿里云提供的海量、安全、低成本、高可靠的云存储服务。支持在任何应用、任何时间、任何地点存储和访问任意类型的数据。
- 阿里云短信服务:覆盖全球的短信服务,友好、高效、智能的互联化通讯能力,帮助企业迅速搭建客户触达通道。
Nacos注册中心
参照文档:Spring Cloud Alibaba Reference Documentation (spring-cloud-alibaba-group.github.io)
参照其中3.3章节
添加依赖:
如果您是 Maven Central 用户,请将我们的 BOM 添加到您的 pom.xml < dependencymanagement > 部分。这将允许您省略任何 Maven 依赖项的版本,而代之以将版本控制委托给 BOM。
<dependencyManagement>
<dependencies>
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
<version>2.1.1.BUILD-SNAPSHOT</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
服务注册/发现: Nacos 发现
写了个
Eureka
注意对比较学习
注册中心原理
在开始搭建 Nacos Discovery 的示例之前,我们先来简单了解下注册中心的原理。
在使用注册中心时,一共有三种角色:服务提供者(Service Provider)、服务消费者(Service Consumer)、注册中心(Registry)。
在一些文章中,服务提供者被称为 Server,服务消费者被称为 Client。胖友们知道即可。
三个角色交互如下图所示:
① Provider:
- 启动时,向 Registry 注册自己为一个服务(Service)的实例(Instance)。
- 同时,定期向 Registry 发送心跳,告诉自己还存活。
- 关闭时,向 Registry 取消注册。
② Consumer:
- 启动时,向 Registry 订阅使用到的服务,并缓存服务的实例列表在内存中。
- 后续,Consumer 向对应服务的 Provider 发起调用时,从内存中的该服务的实例列表选择一个,进行远程调用。
- 关闭时,向 Registry 取消订阅。
③ Registry:
- Provider 超过一定时间未心跳时,从服务的实例列表移除。
- 服务的实例列表发生变化(新增或者移除)时,通知订阅该服务的 Consumer,从而让 Consumer 能够刷新本地缓存。
当然,不同的注册中心可能在实现原理上会略有差异。例如说,Eureka 注册中心,并不提供通知功能,而是 Eureka Client 自己定期轮询,实现本地缓存的更新。
另外,Provider 和 Consumer 是角色上的定义,一个服务同时即可以是 Provider 也可以作为 Consumer。例如说,优惠劵服务可以给订单服务提供接口,同时又调用用户服务提供的接口。
💬:什么?我还是不懂
简单介绍:"买卖"问题
卖家:
快速上手:
仿照
上手
来快速上手
我们来搭建一个 Nacos Discovery 组件的快速入门示例。步骤如下:
- 首先,搭建一个服务提供者
demo-provider
,注册服务到 Nacos 中。 - 然后,搭建一个服务消费者
demo-consumer
,从 Nacos 获取到demo-provider
服务的实例列表,选择其中一个示例,进行 HTTP 远程调用。
注意:版本之间的依赖不要上来就错
1.application.yml
添加Nacos Discovery
组件
spring:
application:
name: demo-provider # Spring 应用名
cloud:
nacos:
# Nacos 作为注册中心的配置项,对应 NacosDiscoveryProperties 配置类
discovery:
server-addr: 127.0.0.1:8848 # Nacos 服务器地址
service: ${spring.application.name} # 注册到 Nacos 的服务名。默认值为 ${spring.application.name}。
server:
port: 8080 # 服务器端口。默认为 8080
看这个就知道Nacos多省事了
接下俩需要配置Nacos看文档配:
2.NacosDiscoveryApplication
添加注册发现组件
@SpringBootApplication
@EnableDiscoveryClient
public class NacosDiscoveryApplication {
public static void main(String[] args) {
SpringApplication.run(NacosDiscoveryApplication.class, args);
}
@RestController
static class TestController {
@GetMapping("/echo")
public String echo(String name) {
return "provider:" + name;
}
}
}
①@SpringBootApplication
注解,被添加在类上,声明这是一个 Spring Boot 应用。Spring Cloud 是构建在 Spring Boot 之上的,所以需要添加。②
@EnableDiscoveryClient
注解,开启 Spring Cloud 的注册发现功能。不过从 Spring Cloud Edgware 版本开始,实际上已经不需要添加@EnableDiscoveryClient
注解,只需要引入 Spring Cloud 注册发现组件,就会自动开启注册发现的功能。例如说,我们这里已经引入了spring-cloud-starter-alibaba-nacos-discovery
依赖,就不用再添加@EnableDiscoveryClient
注解了。
原理介绍:
在 Spring Cloud Common 项目中,定义了 DiscoveryClient 接口,作为通用的发现客户端,提供读取服务和读取服务列表的 API 方法。而想要集成到 Spring Cloud 体系的注册中心的组件,需要提供对应的 DiscoveryClient 实现类
对比记忆:
Spring Cloud Alibaba Nacos Discovery 提供了 NacosDiscoveryClient 实现,Spring Cloud Netflix Eureka 提供了 EurekaDiscoveryClient 实现。
3.运行
出错
原因:版本不一致
不要盲目的
最新
官方推荐使用Hoxton
就用它:
运行:
我就用2021.1
试试看
不给我下载不是我懒🤥
配置买家:
搭建服务消费者
1.配置文件
spring:
application:
name: demo-consumer # Spring 应用名
cloud:
nacos:
# Nacos 作为注册中心的配置项,对应 NacosDiscoveryProperties 配置类
discovery:
server-addr: 127.0.0.1:8848 # Nacos 服务器地址
server:
port: 8081 # 服务器端口。默认为 8080
配置端口必须保持不一致
尽量连续
2.同上开始写启动项目
@SpringBootApplication
public class NacosConsumerApplication {
public static void main(String[] args) {
SpringApplication.run(NacosConsumerApplication.class, args);
}
@Configuration
public class RestTemplateConfiguration {
@Bean
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
@RestController
static class TestController {
@Autowired
private DiscoveryClient discoveryClient;
@Autowired
private RestTemplate restTemplate;
@Autowired
private LoadBalancerClient loadBalancerClient;
@GetMapping("/hello")
public String hello(String name) {
// <1> 获得服务 `demo-provider` 的一个实例
ServiceInstance instance;
if (true) {
// 获取服务 `demo-provider` 对应的实例列表
List<ServiceInstance> instances = discoveryClient.getInstances("demo-provider");
// 选择第一个
instance = instances.size() > 0 ? instances.get(0) : null;
} else {
instance = loadBalancerClient.choose("demo-provider");
}
// <2> 发起调用
if (instance == null) {
throw new IllegalStateException("获取不到实例");
}
String targetUrl = instance.getUri() + "/echo?name=" + name;
String response = restTemplate.getForObject(targetUrl, String.class);
// 返回结果
return "consumer:" + response;
}
}
}
3.运行
@EnableDiscoveryClient
注解,因为已经无需添加,所以我们进行了注释,原因在上面已经解释过。
② RestTemplateConfiguration 配置类,创建 RestTemplate Bean。RestTemplate 是 Spring 提供的 HTTP 调用模板工具类,可以方便我们稍后调用服务提供者的 HTTP API。
③ TestController 提供了 /hello
接口,用于调用服务提供者的 /demo
接口。代码略微有几行,我们来稍微解释下哈。
discoveryClient
属性,DiscoveryClient 对象,服务发现客户端,上文我们已经介绍过。这里我们注入的不是 Nacos Discovery 提供的 NacosDiscoveryClient,保证通用性。未来如果我们不使用 Nacos 作为注册中心,而是使用 Eureka 或则 Zookeeper 时,则无需改动这里的代码。
loadBalancerClient
属性,LoadBalancerClient 对象,负载均衡客户端。稍后我们会使用它,从 Nacos 获取的服务 demo-provider
的实例列表中,选择一个进行 HTTP 调用。
拓展小知识:在 Spring Cloud Common 项目中,定义了 LoadBalancerClient 接口,作为通用的负载均衡客户端,提供从指定服务中选择一个实例、对指定服务发起请求等 API 方法。而想要集成到 Spring Cloud 体系的负载均衡的组件,需要提供对应的 LoadBalancerClient 实现类。例如说,Spring Cloud Netflix Ribbon 提供了 RibbonLoadBalancerClient 实现。
如此,所有需要使用到的地方,只需要获取到 DiscoveryClient 客户端,而无需关注具体实现,保证其通用性。😈 不过貌似 Spring Cloud 体系中,暂时只有 Ribbon 一个负载均衡组件。
当然,LoadBalancerClient 的服务的实例列表,是来自 DiscoveryClient 提供的。
/hello
接口,示例接口,对服务提供者发起一次 HTTP 调用。
<1>
处,获得服务demo-provider
的一个实例。这里我们提供了两种方式的代码,分别基于 DiscoveryClient 和 LoadBalancerClient。<2>
处,通过获取到的服务实例 ServiceInstance 对象,拼接请求的目标 URL,之后使用 RestTemplate 发起 HTTP 调用。
实验后:
访问[127.0.0.1:8081/hello?name=yudaoyuanma]
远掉成功!!
Nacos配置中心
推荐官方文档
Nacos 配置快速访问 Nacos 组态管理能力基于 Spring Cloud 的编程模型。
github
官方推荐:(盖章基于该文档)Nacos config · alibaba/spring-cloud-alibaba Wiki (github.com)
简单介绍
1.进入配置中心进行手动配置
2.完善相关的操作
出现这个之前要在pom中配置的
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<excludes>
<exclude>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
</exclude>
</excludes>
</configuration>
</plugin>
</plugins>
</build>
主要是自动装配失效
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
支持动态配置:
Nacos Config 也支持动态配置更新。开始 Spring Boot 应用程序测试的代码如下:
@SpringBootApplication
public class NacosConfigApplication {
public static void main(String[] args) {
ConfigurableApplicationContext applicationContext = SpringApplication.run(NacosConfigApplication.class, args);
while(true) {
//When configurations are refreshed dynamically, they will be updated in the Enviroment, therefore here we retrieve configurations from Environment every other second.
String userName = applicationContext.getEnvironment().getProperty("user.name");
String userAge = applicationContext.getEnvironment().getProperty("user.age");
System.err.println("user name :" + userName + "; age: " + userAge);
TimeUnit.SECONDS.sleep(1);
}
}
}
更改 user.name 时,可以从应用程序中检索最新的值,如下所示:
user name :nacos-config-yaml; age: 68user name :nacos-config-yaml; age: 68user name :nacos-config-yaml; age: 682018-11-02 15:04:25.069 INFO 32957 --- [-127.0.0.1:8848] o.s.boot.SpringApplication : Started application in 0.144 seconds (JVM running for 71.752)2018-11-02 15:04:25.070 INFO 32957 --- [-127.0.0.1:8848] s.c.a.AnnotationConfigApplicationContext : Closing org.springframework.context.annotation.AnnotationConfigApplicationContext@10c89124: startup date [Fri Nov 02 15:04:25 CST 2018]; parent: org.springframework.context.annotation.AnnotationConfigApplicationContext@6520af72018-11-02 15:04:25.071 INFO 32957 --- [-127.0.0.1:8848] s.c.a.AnnotationConfigApplicationContext : Closing org.springframework.context.annotation.AnnotationConfigApplicationContext@6520af7: startup date [Fri Nov 02 15:04:24 CST 2018]; root of context hierarchy//Read the updated value from Enviromentuser name :nacos-config-yaml-update; age: 68user name :nacos-config-yaml-update; age: 68
了解:
1.优势:
Nacos 提供用于存储配置和其他元数据的 key/value 存储,为分布式系统中的外部化配置提供服务器端和客户端支持。使用 Spring Cloud Alibaba Nacos Config,您可以在 Nacos Server 集中管理你 Spring Cloud 应用的外部属性配置。
2.为了什么问题:
Spring Cloud Alibaba Nacos Config 是 Config Server 和 Client 的替代方案,客户端和服务器上的概念与 Spring Environment 和 PropertySource 有着一致的抽象,在特殊的 bootstrap 阶段,配置被加载到 Spring 环境中。当应用程序通过部署管道从开发到测试再到生产时,您可以管理这些环境之间的配置,并确保应用程序具有迁移时需要运行的所有内容。
上手:(仔细看官方文档)
必须使用 bootstrap.properties 配置文件来配置Nacos Server 地址,例如:
bootstrap.properties
spring.application.name=nacos-configspring.cloud.nacos.config.server-addr=127.0.0.1:8848
配置文件事项
创建 bootstrap.yaml
配置文件,添加 Nacos Config 相关配置。配置如下:
spring: application: name: demo-application cloud: nacos: # Nacos Config 配置项,对应 NacosConfigProperties 配置属性类 config: server-addr: 127.0.0.1:8848 # Nacos 服务器地址 namespace: # 使用的 Nacos 的命名空间,默认为 null group: DEFAULT_GROUP # 使用的 Nacos 配置分组,默认为 DEFAULT_GROUP name: # 使用的 Nacos 配置集的 dataId,默认为 spring.application.name file-extension: yaml # 使用的 Nacos 配置集的 dataId 的文件拓展名,同时也是 Nacos 配置集的配置格式,
注意:为什么要新建一个
bootstrap.yaml
? 因为会首先加载这个
思考
🤿:为什么需要多一个这个配置文件呢
在 Spring Cloud 应用中,会先创建一个 Bootstrap Context(引导上下文),比 Spring Boot 创建 Application Context(应用上下文)更早初始化。
Bootstrap Context 新增了一个 bootstrap.yaml
配置文件,保证和 Application Context 的 application.yaml
配置文件的隔离。
有了配置文件的隔离之后,Bootstrap Context 初始化的 Bean 从哪里来?Spring Cloud 新定义了专属于 Bootstrap Context 的自动化配置类的拓展点 BootstrapConfiguration,和 Spring Boot 为 Application Context 的自动化配置类的拓展点 EnableAutoConfiguration的隔离,保证两个 Context 创建各自的 Bean。以 Spring Cloud Alibaba Nacos 的 spring.factories
举例子,如下图所示:
虽然说,Bootstrap Context 和 Application Context 做了这么多隔离,但是它们有一点是共享的,那就是 Environment。在 Spring 中,我们通过 Environment 获取属性配置,例如说 spring.application.name
对应的值是多少。
了解完这些之后,我们把它们串联在一起去思考一下,Bootstrap Context 的目的究竟是什么呢?通过 Bootstrap Context 的优先初始化,将配置加载到 Environment 中,提供给后面的 Application Context 使用。
举个贼重要的例子,稍后我们会在 bootstrap.yaml
添加 Spring Cloud Alibaba Nacos Config 相关的配置,这样 Bootstrap Context 在初始化时,通过 NacosConfigBootstrapConfiguration 创建 Nacos 相关的 Bean,然后实现从 Nacos 配置中心加载配置到 Environment 中。
如果我们把 Spring Cloud Alibaba Nacos Config 相关的配置添加在 application.yaml
中,那么可能无法保证 Nacos 相关的 Bean 被最先初始化,完成从 Nacos 获取配置,从而影响创建的 Bean。
友情提示:实际上将 Spring Cloud Alibaba Nacos Config 相关的配置添加在application.yaml
中,也有办法解决,需要基于 Spring Boot 的 ApplicationContextInitializer 和 EnvironmentPostProcessor 拓展点,实现自定义的处理。感兴趣的胖友,可以去看看 Apollo 或者 Nacos Config Spring Boot Starter 的源码。
🎇详细解释Nacos Config 配置项
Nacos Config 配置项,以 spring.cloud.nacos.config
开头,对应 NacosConfigProperties 配置属性类。
① server-addr
配置项,设置 Nacos 服务器地址。
② namespace
配置项,使用的 Nacos 的命名空间,默认为 null
,表示使用 public
这个默认命名空间。
FROM 《Nacos 文档 —— Nacos 概念》命名空间
用于进行租户粒度的配置隔离。不同的命名空间下,可以存在相同的 Group 或 Data ID 的配置。Namespace 的常用场景之一是不同环境的配置的区分隔离,例如开发测试环境和生产环境的资源(如配置、服务)隔离等。
③ group
配置项,使用的 Nacos 配置分组,默认为 DEFAULT_GROUP
。
FROM 《Nacos 文档 —— Nacos 概念》配置分组
Nacos 中的一组配置集,是组织配置的维度之一。通过一个有意义的字符串(如 Buy 或 Trade )对配置集进行分组,从而区分 Data ID 相同的配置集。当您在 Nacos 上创建一个配置时,如果未填写配置分组的名称,则配置分组的名称默认采用DEFAULT_GROUP
。配置分组的常见场景:不同的应用或组件使用了相同的配置类型,如database_url
配置和MQ_topic
配置。
④ name
配置项,使用的 Nacos 配置集的 dataId,默认为 spring.application.name
。
FROM 《Nacos 文档 —— Nacos 概念》配置集
一组相关或者不相关的配置项的集合称为配置集。在系统中,一个配置文件通常就是一个配置集,包含了系统各个方面的配置。例如,一个配置集可能包含了数据源、线程池、日志级别等配置项。配置集 ID
Nacos 中的某个配置集的 ID。配置集 ID 是组织划分配置的维度之一。Data ID 通常用于组织划分系统的配置集。一个系统或者应用可以包含多个配置集,每个配置集都可以被一个有意义的名称标识。Data ID 通常采用类 Java 包(如com.taobao.tc.refund.log.level
)的命名规则保证全局唯一性。此命名规则非强制。
因为这里我们未进行配置,所以使用 Nacos 配置集的 dataId 为 demo-application
。这也是为什么我们将 spring.application.name
配置项添加到 bootstrap.yaml
配置文件中的原因。
⑤ file-extension
配置项,使用的 Nacos 配置集的 dataId 的文件拓展名,同时也是 Nacos 配置集的配置格式,默认为 properties
。这里我们设置为 yaml
,因为我们稍后使用的配置集的配置格式为 YAML
。
Nacos多环境 配置集
看官方怎么做使用 Nacos Config 加载配置时,使用 DataId 的基本配置为
${ spring.application.name }
。${ file-extension: properties }
,和DataId
的${ spring.application.name }-${ profile }
。${ file-extension: properties }
也被加载。如果您需要在不同的环境中使用不同的配置,您可以使用Spring
提供的${ Spring.profiles.active }
配置。
:thinking:这是什么意思呢?
换句话说,我们开发时需要在配置列表中新建配置,而现在只需要在配置文件bootstrap
中配置
Data ID: nacos-config-develop.yamlGroup : DEFAULT_GROUPConfiguration format: YAMLConfiguration content: current.env: develop-env
看看启动项里面的内容变化:
@SpringBootApplicationpublic class NacosConfigApplication { public static void main(String[] args) { ConfigurableApplicationContext applicationContext = SpringApplication.run(NacosConfigApplication.class, args); while(true) { String userName = applicationContext.getEnvironment().getProperty("user.name"); String userAge = applicationContext.getEnvironment().getProperty("user.age"); //Get the current deployment environment String currentEnv = applicationContext.getEnvironment().getProperty("current.env"); System.err.println("in "+currentEnv+" enviroment; "+"user name :" + userName + "; age: " + userAge); TimeUnit.SECONDS.sleep(1); } }}
String currentEnv = applicationContext.getEnvironment().getProperty("current.env");
这句话表示这个是个开发环境develop-env
运行结果
in develop-env enviroment; user name :nacos-config-yaml-update; age: 682018-11-02 15:34:25.013 INFO 33014 --- [ Thread-11] ConfigServletWebServerApplicationContext : Closing
切换到生产环境:
同理仅仅改变 ${ spring.profiles.active }
的参数
spring.profiles.active=product
🥶官方提示!!
我们使用spring.profiles.active = < profileename >
方法在配置文件中对配置进行了编码。在真实的场景中,这个变量需要在不同的环境中有所不同。您可以使用-Dspring.profiles.active = < profile >
参数来指定配置,这样您就可以轻松地在不同的环境之间切换。
Nacos自定义namespace(命名空间)
🚩推荐使用
名称空间用于隔离不同租户的配置。组和数据 id 可以在不同的名称空间中相同。名称空间的典型场景是不同环境的配置隔离,例如,开发/测试环境和生产环境(配置和服务等)之间的隔离。
推荐:labx-05-sca-nacos-config-demo-profiles。可以下载博客链接芋道 Spring Cloud Alibaba 配置中心 Nacos 入门 | 芋道源码 —— 纯源码解析博客 (iocoder.cn)第3.多环境配置这里很详细
1.配置的时候ID
在这里:
2.创建配置集
① 打开 Nacos UI 界面的「配置列表」菜单,进入「配置管理」功能。如下图所示:
② 点击 dev
命名空间,然后点击列表右上角的➕号,进入「新建配置」界面,创建一个 Nacos 配置集。输入如下内容,并点击「发布」按钮,完成创建。如下图所示:
③ 点击 prod
命名空间,然后点击列表右上角的➕号,进入「新建配置」界面,创建一个 Nacos 配置集。输入如下内容,并点击「发布」按钮,完成创建。如下图所示:
3.编码
略
4.运行
下面,我们使用命令行参数进行 --spring.profiles.active
配置项,实现不同环境,读取不同配置文件。
经过测试:
- 使用命令行参数进行
--spring.profiles.active
配置,对bootstrap.yaml
配置文件无效。- 使用 VM 参数进行
-Dspring.profiles.active
配置爱,对bootstrap.yaml
配置文件有效。具体的原因还不知道,先暂时这么解决哈~
① 开发环境示例:直接在 IDEA 中,增加 -Dspring.profiles.active=dev
到 VM options 中。如下图所示:
Nacos自定义(group)群组
默认情况下,当没有定义{ spring.cloud.nacos.config.group }
配置时,使用 DEFAULT _ group
。如果您需要定义自己的组,可以在以下属性中定义它:
spring.cloud.nacos.config.group=DEVELOP_GROUP
这个配置必须在 bootstrap.properties 文件中,Group 的值必须与 spring.cloud.nacos.config.Group 的值相同。
Nacos自定义Data Id(数据ID) 配置
完整配置:
spring.application.name=opensource-service-providerspring.cloud.nacos.config.server-addr=127.0.0.1:8848# config external configuration# 1. Data Id is in the default group of DEFAULT_GROUP, and dynamic refresh of configurations is not supported.spring.cloud.nacos.config.ext-config[0].data-id=ext-config-common01.properties# 2. Data Id is not in the default group, and dynamic refresh of configurations is not supported.spring.cloud.nacos.config.ext-config[1].data-id=ext-config-common02.propertiesspring.cloud.nacos.config.ext-config[1].group=GLOBALE_GROUP# 3. Data Id is not in the default group and dynamic referesh of configurations is supported.spring.cloud.nacos.config.ext-config[2].data-id=ext-config-common03.propertiesspring.cloud.nacos.config.ext-config[2].group=REFRESH_GROUPspring.cloud.nacos.config.ext-config[2].refresh=true
通过配置spring.cloud.nacos.config.ext-config [ n ] . data-id
来支持多个数据 id通过配置
spring.cloud.nacos.config.ext-config [ n ] . group
来自定义数据 id 组。通过配置
spring.cloud.nacos.config.ext-config [ n ]
,控制配置更改时是否支持此数据 id 支持动态刷新配置。刷新。默认情况下不支持它。
Note 多个 Data Id 同时配置时,他的优先级关系是 spring.cloud.nacos.config.extension-configs[n].data-id
其中 n 的值越大,优先级越高。
Note spring.cloud.nacos.config.extension-configs[n].data-id
的值必须带文件扩展名,文件扩展名既可支持 properties,又可以支持 yaml/yml。 此时spring.cloud.nacos.config.file-extension
的配置对自定义扩展配置的 Data Id 文件扩展名没有影响。 更加清晰的在多个应用间配置共享的 Data Id ,你可以通过以下的方式来配置:# 配置支持共享的 Data Idspring.cloud.nacos.config.shared-configs[0].data-id=common.yaml# 配置 Data Id 所在分组,缺省默认 DEFAULT_GROUPspring.cloud.nacos.config.shared-configs[0].group=GROUP_APP1# 配置Data Id 在配置变更时,是否动态刷新,缺省默认 falsespring.cloud.nacos.config.shared-configs[0].refresh=true
可以看到:
- 通过
spring.cloud.nacos.config.shared-configs[n].data-id
来支持多个共享 Data Id 的配置。- 通过
spring.cloud.nacos.config.shared-configs[n].group
来配置自定义 Data Id 所在的组,不明确配置的话,默认是 DEFAULT_GROUP。- 通过
spring.cloud.nacos.config.shared-configs[n].refresh
来控制该Data Id在配置变更时,是否支持应用中动态刷新,默认false。>配置的优先级
Spring Cloud Alibaba Nacos Config 目前提供了三种配置能力从 Nacos 拉取相关的配置。
- A: 通过
spring.cloud.nacos.config.shared-configs[n].data-id
支持多个共享 Data Id 的配置- B: 通过
spring.cloud.nacos.config.extension-configs[n].data-id
的方式支持多个扩展 Data Id 的配置- C: 通过内部相关规则(应用名、应用名+ Profile )自动生成相关的 Data Id 配置
当三种方式共同使用时,他们的一个优先级关系是:A < B < C
自定义 Data Id 的详细设计
关于这部分的详细设计:
SCA Nacos Config 共享配置方案设计
随着社区的回馈,发现 Spring Cloud Alibaba Nacos Config Starter 目前不能完美的来支持多个应用间的一些共享配置。
在实际的业务场景中应用和共享配置间的关系可能如下图所示:
-
从单个应用的角度来看: 应用可能会有多套(develop/beta/product)发布环境,多套发布环境之间有不同的基础配置,例如数据库。
-
目前 SCA Nacos Config 可以灵活的支持单个应用间在多套环境可灵活的切换,但是在多个应用间共享一些通用的配置支持的还不是很完美。
本方案设计的目标就是来解决这个问题。目前有三种设计方案,分别是: -
新增一个应用分组的配置,分组命名的格式是 通过域名的命名方式,来自动的生成共享配置的 Data Id。
-
通过类似面向对象的方式自定义配置(相对于第二种的升级版)。
下面分别来分析三种方案的具体实现和优缺点。
基于域名的配置方式给应用分组
通过一个配置参数(例如:${spring.application.group}) 来指明当前应用所属的分组(或者说所属的域)。
例如我有两个应用分别为Order_Application和Auth_Application,给这两个应用配置的分组名(域)是:
spring.application.group=com.alibaba.aliware.edas
那对于Spring Cloud Alibaba Nacos Config 来说,多个应用可以属于 com.alibaba 这个应用分组(域),也可以属于
com.alibaba.aliware 这个应用分组(域),当然也可以属于 com.alibaba.aliware.edas 这个应用分组(域)。
罗里吧嗦了这么多,目的就是 Data Id 通过以这个分组(域)来命名,从而实现多个应用间在某个分组(域)下的共享配置。如下图所示:以这种方式来实现多个应用间的配置共享,可以看出他具有天然的局限性。
-
受到 ${spring.application.group }配置的影响,Data Id 的表现力是非常有限的(当然85%的场景应该够用了)。
-
一方面是Data Id 的命名个数受到了限制。
-
另一方面如果两个应用的 file-extension 不一致(一个是properties,一个是yaml),那这个时候共享配置的Data Id 必须同时要含有 properties 和 yaml 为扩展名的配置。
-
Data Id 的命名也受到了 ${spring.application.group } 配置的束缚。
-
-
学习成本偏高(当然还是可以学会的)。学习成本偏高体现在:
-
要知道Data Id 的命名规则,才能在项目实施过程中对 Data Id 的命名运用自如。
-
此外对于域名命名的层次个数也不太好把握。少了的话,担心dataid的个数不够用,多了的话看上去有显得的比较冗余。
-
还需要学习并理解这里配置的优先级,不然在程序中有可能就会拿到意想不到的配置。
-
-
易出错(当然是可以克服的)。对于多级应用分组的配置共享,这个时候Data Id 的命名要格外注意了。注意他们的层次关系,Data Id 书写时不要张冠李戴。
-
实现起来稍微复杂
当然他的好处也非常明显,当你理解了他背后的设计理念时,这个共享配置的层次也非常明显。因为层次的关系天然依托于域名的层次关系。
自定义的方式来命名 Data Id
这种方式实现简单易懂,即 SCA Nacos Config 会新增加一个配置,用来配置可实现共享配置所有的 Data Id。如下所示:
spring.cloud.nacos.shared.dataids=global.yaml,app-common.yaml,app-local-common.yaml
NOTE: 为了尽可能的和Nacos使用方式(即Data Id 是一个带有额外文件扩展名的)保持一致,这里配置的Data Id 是一定需要带上文件扩展名的。这个时候两个应用(或多个应用)之间共享配置的 Data Id 关系如下图所示:
Spring Boot 提倡约定大于配置。当使用这种方式来实现应用间的共享配置时,我们也继承了Spring Boot的这个优良传统,多个共享配置间的一个优先级的关系我们约定:按照配置出现的先后顺序,即后面的优先级要高于前面的。
这种方式的优点在于:
-
dataid的命名方式完全交给业务方本身,不受 SCA Nacos Config Starter 实现的束缚。
-
dataid的命名方式既可以参考第一种方式来命名,又可以充分的发挥主观能动性,结合自己实际的业务给dataid命名。
-
减少了多个应用间如果file-extension不一致,为每个 file-extension 多加这么一个配置的麻烦。
-
当使用这种方式时,不会为这些共享配置强制绑定一个 file-extenson,即可以直接在我们暴露出来的一个变量中 dataid以file-extension 结尾。如果没有显示的说明,这个时候就会以file-extenson为准。
当然这种方案的缺点在于扩展性不强。即如果对于某个共享配置需要做额外的配置,例如额外配置Group/是否需要刷新/是否需要从本地缓存加载等等。因此为了应对这种类型的场景,小组内讨论出了第三种方案。
通过类似面向对象方式的自定义配置
说明: Spring 可以支持在加了 ConfigurationProperties 注解配置类的内部某个对象实例来注入应用中的一些配置。这种使用方式查了一下官方和网络上没有一个大的标题总结,结合这种方式很像给某个实例中的字段赋值,所以这里先暂时取名:
类似面向对象方式的自定义配置(有更好能够形象的标明其含义的命名可以在下面留下评论-_-)。这种方案沿用了第二种设计方案的优点,同时又弥补了第二种方案的不足。我们通过内部定义一个对象,来支持一些灵活的扩展配置。
我们给这个对象可以预留一些可扩展的配置字段。例如:
public class Config{ private String dataId; private String group = "DEFAULT_GROUP"; private Boolean refresh = false; //.....后期可能还有其他的一些配置 //省略 set/get 方法}
最终在实现时是可以支持以list的方式来配置其值。如下是两个扩展配置的实例:
spring.cloud.nacos.config.ext-config[0].data-id=global-shared.properties # group 和referesh 使用默认值spring.cloud.nacos.config.ext-config[1].data-id=app-common.propertiesspring.cloud.nacos.config.ext-config[1].group=DEVELOP_GROUP #配置自定义所在的组spring.cloud.nacos.config.ext-config[1].refresh=true #需要刷新
NOTE: 为了尽可能的和Nacos使用方式(即data id是一个带有额外文件扩展名的)保持一致,这里配置的dataid是一定需要带上文件扩展名的。
最终的实现
SCA Nacos Config 在第二种方案和第三种方案的实现上是并存的。如果你觉得第三种方案配置的比较麻烦,同时第二种方案就可以满足你的需求,这个时候就可以选择第二种方案。
如果你需外可读性好、层级感比较明显、后期的扩展性更强,那这个时候第三种方案也是OK的。
Nacos概念原理
完全参照:芋道 Spring Cloud Alibaba 注册中心 Nacos 入门 | 芋道源码 —— 纯源码解析博客 (iocoder.cn)
基本架构及概念
服务 (Service)
服务是指一个或一组软件功能(例如特定信息的检索或一组操作的执行),其目的是不同的客户端可以为不同的目的重用(例如通过跨进程的网络调用)。Nacos 支持主流的服务生态,如 Kubernetes Service、gRPC|Dubbo RPC Service 或者 Spring Cloud RESTful Service。
服务注册中心 (Service Registry)
服务注册中心,它是服务,其实例及元数据的数据库。服务实例在启动时注册到服务注册表,并在关闭时注销。服务和路由器的客户端查询服务注册表以查找服务的可用实例。服务注册中心可能会调用服务实例的健康检查 API 来验证它是否能够处理请求。
服务元数据 (Service Metadata)
服务元数据是指包括服务端点(endpoints)、服务标签、服务版本号、服务实例权重、路由规则、安全策略等描述服务的数据。
服务提供方 (Service Provider)
是指提供可复用和可调用服务的应用方。
服务消费方 (Service Consumer)
是指会发起对某个服务调用的应用方。
配置 (Configuration)
在系统开发过程中通常会将一些需要变更的参数、变量等从代码中分离出来独立管理,以独立的配置文件的形式存在。目的是让静态的系统工件或者交付物(如 WAR,JAR 包等)更好地和实际的物理运行环境进行适配。配置管理一般包含在系统部署的过程中,由系统管理员或者运维人员完成这个步骤。配置变更是调整系统运行时的行为的有效手段之一。
配置管理 (Configuration Management)
在数据中心中,系统中所有配置的编辑、存储、分发、变更管理、历史版本管理、变更审计等所有与配置相关的活动统称为配置管理。
名字服务 (Naming Service)
提供分布式系统中所有对象(Object)、实体(Entity)的“名字”到关联的元数据之间的映射管理服务,例如 ServiceName -> Endpoints Info, Distributed Lock Name -> Lock Owner/Status Info, DNS Domain Name -> IP List, 服务发现和 DNS 就是名字服务的2大场景。
配置服务 (Configuration Service)
在服务或者应用运行过程中,提供动态配置或者元数据以及配置管理的服务提供者。
更多概念...
逻辑架构及其组件介绍
- 服务管理:实现服务CRUD,域名CRUD,服务健康状态检查,服务权重管理等功能
- 配置管理:实现配置管CRUD,版本管理,灰度管理,监听管理,推送轨迹,聚合数据等功能
- 元数据管理:提供元数据CURD 和打标能力
- 插件机制:实现三个模块可分可合能力,实现扩展点SPI机制
- 事件机制:实现异步化事件通知,sdk数据变化异步通知等逻辑
- 日志模块:管理日志分类,日志级别,日志可移植性(尤其避免冲突),日志格式,异常码+帮助文档
- 回调机制:sdk通知数据,通过统一的模式回调用户处理。接口和数据结构需要具备可扩展性
- 寻址模式:解决ip,域名,nameserver、广播等多种寻址模式,需要可扩展
- 推送通道:解决server与存储、server间、server与sdk间推送性能问题
- 容量管理:管理每个租户,分组下的容量,防止存储被写爆,影响服务可用性
- 流量管理:按照租户,分组等多个维度对请求频率,长链接个数,报文大小,请求流控进行控制
- 缓存机制:容灾目录,本地缓存,server缓存机制。容灾目录使用需要工具
- 启动模式:按照单机模式,配置模式,服务模式,dns模式,或者all模式,启动不同的程序+UI
- 一致性协议:解决不同数据,不同一致性要求情况下,不同一致性机制
- 存储模块:解决数据持久化、非持久化存储,解决数据分片问题
- Nameserver:解决namespace到clusterid的路由问题,解决用户环境与nacos物理环境映射问题
- CMDB:解决元数据存储,与三方cmdb系统对接问题,解决应用,人,资源关系
- Metrics:暴露标准metrics数据,方便与三方监控系统打通
- Trace:暴露标准trace,方便与SLA系统打通,日志白平化,推送轨迹等能力,并且可以和计量计费系统打通
- 接入管理:相当于阿里云开通服务,分配身份、容量、权限过程
- 用户管理:解决用户管理,登录,sso等问题
- 权限管理:解决身份识别,访问控制,角色管理等问题
- 审计系统:扩展接口方便与不同公司审计系统打通
- 通知系统:核心数据变更,或者操作,方便通过SMS系统打通,通知到对应人数据变更
- OpenAPI:暴露标准Rest风格HTTP接口,简单易用,方便多语言集成
- Console:易用控制台,做服务管理、配置管理等操作
- SDK:多语言sdk
- Agent:dns-f类似模式,或者与mesh等方案集成
- CLI:命令行对产品进行轻量化管理,像git一样好用
领域模型
数据模型
Nacos 数据模型 Key 由三元组唯一确定, Namespace默认是空串,公共命名空间(public),分组默认是 DEFAULT_GROUP。
服务领域模型
配置领域模型
围绕配置,主要有两个关联的实体,一个是配置变更历史,一个是服务标签(用于打标分类,方便索引),由 ID 关联。
类视图
Nacos-SDK 类视图
服务部分待续
构建物、部署及启动模式
两种交付工件
Nacos 支持标准 Docker 镜像(TODO: 0.2版本开始支持)及 zip(tar.gz)压缩包的构建物。
两种启动模式
Nacos 支持将注册中心(Service Registry)与配置中心(Config Center) 在一个进程合并部署或者将2者分离部署的两种模式。
免费的公有云服务模式
除了您自己部署和启动 Nacos 服务之外,在云计算时代,Nacos 也支持公有云模式,在阿里云公有云的商业产品(如ACM, EDAS) 中会提供 Nacos 的免费的公有云服务。我们也欢迎和支持其他的公有云提供商提供 Nacos 的公有云服务。
Nacos 多环境配置概念
NOTE: Nacos 引入了一些基本的概念,系统性的了解一下这些概念可以帮助您更好的理解和正确的使用 Nacos 产品。
地域
物理的数据中心,资源创建成功后不能更换。
可用区
同一地域内,电力和网络互相独立的物理区域。同一可用区内,实例的网络延迟较低。
接入点
地域的某个服务的入口域名。
命名空间
用于进行租户粒度的配置隔离。不同的命名空间下,可以存在相同的 Group 或 Data ID 的配置。Namespace 的常用场景之一是不同环境的配置的区分隔离,例如开发测试环境和生产环境的资源(如配置、服务)隔离等。
配置
在系统开发过程中,开发者通常会将一些需要变更的参数、变量等从代码中分离出来独立管理,以独立的配置文件的形式存在。目的是让静态的系统工件或者交付物(如 WAR,JAR 包等)更好地和实际的物理运行环境进行适配。配置管理一般包含在系统部署的过程中,由系统管理员或者运维人员完成。配置变更是调整系统运行时的行为的有效手段。
配置管理
系统配置的编辑、存储、分发、变更管理、历史版本管理、变更审计等所有与配置相关的活动。
配置项
一个具体的可配置的参数与其值域,通常以 param-key=param-value 的形式存在。例如我们常配置系统的日志输出级别(logLevel=INFO|WARN|ERROR) 就是一个配置项。
配置集
一组相关或者不相关的配置项的集合称为配置集。在系统中,一个配置文件通常就是一个配置集,包含了系统各个方面的配置。例如,一个配置集可能包含了数据源、线程池、日志级别等配置项。
配置集 ID
Nacos 中的某个配置集的 ID。配置集 ID 是组织划分配置的维度之一。Data ID 通常用于组织划分系统的配置集。一个系统或者应用可以包含多个配置集,每个配置集都可以被一个有意义的名称标识。Data ID 通常采用类 Java 包(如 com.taobao.tc.refund.log.level)的命名规则保证全局唯一性。此命名规则非强制。
配置分组
Nacos 中的一组配置集,是组织配置的维度之一。通过一个有意义的字符串(如 Buy 或 Trade )对配置集进行分组,从而区分 Data ID 相同的配置集。当您在 Nacos 上创建一个配置时,如果未填写配置分组的名称,则配置分组的名称默认采用 DEFAULT_GROUP 。配置分组的常见场景:不同的应用或组件使用了相同的配置类型,如 database_url 配置和 MQ_topic 配置。
配置快照
Nacos 的客户端 SDK 会在本地生成配置的快照。当客户端无法连接到 Nacos Server 时,可以使用配置快照显示系统的整体容灾能力。配置快照类似于 Git 中的本地 commit,也类似于缓存,会在适当的时机更新,但是并没有缓存过期(expiration)的概念。
服务
通过预定义接口网络访问的提供给客户端的软件功能。
服务名
服务提供的标识,通过该标识可以唯一确定其指代的服务。
服务注册中心
存储服务实例和服务负载均衡策略的数据库。
服务发现
在计算机网络上,(通常使用服务名)对服务下的实例的地址和元数据进行探测,并以预先定义的接口提供给客户端进行查询。
元信息
Nacos数据(如配置和服务)描述信息,如服务版本、权重、容灾策略、负载均衡策略、鉴权配置、各种自定义标签 (label),从作用范围来看,分为服务级别的元信息、集群的元信息及实例的元信息。
应用
用于标识服务提供方的服务的属性。
服务分组
不同的服务可以归类到同一分组。
虚拟集群
同一个服务下的所有服务实例组成一个默认集群, 集群可以被进一步按需求划分,划分的单位可以是虚拟集群。
实例
提供一个或多个服务的具有可访问网络地址(IP:Port)的进程。
权重
实例级别的配置。权重为浮点数。权重越大,分配给该实例的流量越大。
健康检查
以指定方式检查服务下挂载的实例 (Instance) 的健康度,从而确认该实例 (Instance) 是否能提供服务。根据检查结果,实例 (Instance) 会被判断为健康或不健康。对服务发起解析请求时,不健康的实例 (Instance) 不会返回给客户端。
健康保护阈值
为了防止因过多实例 (Instance) 不健康导致流量全部流向健康实例 (Instance) ,继而造成流量压力把健康实例 (Instance) 压垮并形成雪崩效应,应将健康保护阈值定义为一个 0 到 1 之间的浮点数。当域名健康实例数 (Instance) 占总服务实例数 (Instance) 的比例小于该值时,无论实例 (Instance) 是否健康,都会将这个实例 (Instance) 返回给客户端。这样做虽然损失了一部分流量,但是保证了集群中剩余健康实例 (Instance) 能正常工作。
Sentinel服务容错
先开始官方文档的学习在实践上手
1.Sentinel简介
分布式系统的流量防卫兵
Sentinel 具有以下特征:
- 丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围)、消息削峰填谷、集群流量控制、实时熔断下游不可用应用等。
- 完备的实时监控:Sentinel 同时提供实时的监控功能。您可以在控制台中看到接入应用的单台机器秒级数据,甚至 500 台以下规模的集群的汇总运行情况。
- 广泛的开源生态:Sentinel 提供开箱即用的与其它开源框架/库的整合模块,例如与 Spring Cloud、Apache Dubbo、gRPC、Quarkus 的整合。您只需要引入相应的依赖并进行简单的配置即可快速地接入 Sentinel。同时 Sentinel 提供 Java/Go/C++ 等多语言的原生实现。
- 完善的 SPI 扩展机制:Sentinel 提供简单易用、完善的 SPI 扩展接口。您可以通过实现扩展接口来快速地定制逻辑。例如定制规则管理、适配动态数据源等。
Sentinel 的主要特性:
Sentinel 的开源生态:
Sentinel 分为两个部分:
- 核心库(Java 客户端)不依赖任何框架/库,能够运行于所有 Java 运行时环境,同时对 Dubbo / Spring Cloud 等框架也有较好的支持。
- 控制台(Dashboard)基于 Spring Boot 开发,打包后可以直接运行,不需要额外的 Tomcat 等应用容器。
Sentinel 控制台
1. 概述
Sentinel 提供一个轻量级的开源控制台,它提供机器发现以及健康情况管理、监控(单机和集群),规则管理和推送的功能。这里,我们将会详细讲述如何通过简单的步骤就可以使用这些功能。
接下来,我们将会逐一介绍如何整合 Sentinel 核心库和 Dashboard,让它发挥最大的作用。同时我们也在阿里云上提供企业级的 Sentinel 服务:AHAS Sentinel 控制台,您只需要几个简单的步骤,就能最直观地看到控制台如何实现这些功能,并体验多样化的监控及全自动托管的集群流控能力。
Sentinel 控制台包含如下功能:
- 查看机器列表以及健康情况:收集 Sentinel 客户端发送的心跳包,用于判断机器是否在线。
- 监控 (单机和集群聚合):通过 Sentinel 客户端暴露的监控 API,定期拉取并且聚合应用监控信息,最终可以实现秒级的实时监控。
- 规则管理和推送:统一管理推送规则。
- 鉴权:生产环境中鉴权非常重要。这里每个开发者需要根据自己的实际情况进行定制。
正式学习:
我也不是很会,好在时中文的文档参考;
新手指南 · alibaba/Sentinel Wiki (github.com)
介绍 · alibaba/Sentinel Wiki (github.com)
控制台 · alibaba/Sentinel Wiki (github.com)
适用于组件
sentinel的工作原理:
该部分半分析官方文档
流程:
在 Sentinel 里面,所有的资源都对应一个资源名称(
resourceName
),每次资源调用都会创建一个Entry
对象。Entry 可以通过对主流框架的适配自动创建,也可以通过注解的方式或调用SphU
API 显式创建。Entry 创建的时候,同时也会创建`一系列功能插槽
(slot chain
)
NodeSelectorSlot
负责收集资源的路径,并将这些资源的调用路径,以树状结构存储起来,用于根据调用路径来限流降级;ClusterBuilderSlot
则用于存储资源的统计信息以及调用者信息,例如该资源的 RT, QPS, thread count 等等,这些信息将用作为多维度限流,降级的依据;StatisticSlot
则用于记录、统计不同纬度的 runtime 指标监控信息;FlowSlot
则用于根据预设的限流规则以及前面 slot 统计的状态,来进行流量控制;AuthoritySlot
则根据配置的黑白名单和调用来源信息,来做黑白名单控制;DegradeSlot
则通过统计信息以及预设的规则,来做熔断降级;SystemSlot
则通过系统的状态,例如 load1 等,来控制总的入口流量;核心类解释如下:(里面的太简单了)
ProcessorSlotChain
Sentinel 的核心骨架,将不同的 Slot 按照顺序串在一起(责任链模式),从而将不同的功能(限流、降级、系统保护)组合在一起。slot chain 其实可以分为两部分:统计数据构建部分(statistic)和判断部分(rule checking)。核心结构:
目前的设计是 one slot chain per resource,因为某些 slot 是 per resource 的(比如 NodeSelectorSlot)。
Context
Context 代表调用链路上下文,贯穿一次调用链路中的所有
Entry
。Context 维持着入口节点(entranceNode
)、本次调用链路的 curNode、调用来源(origin
)等信息。Context 名称即为调用链路入口名称。Context 维持的方式:通过 ThreadLocal 传递,只有在入口
enter
的时候生效。由于 Context 是通过 ThreadLocal 传递的,因此对于异步调用链路,线程切换的时候会丢掉 Context,因此需要手动通过ContextUtil.runOnContext(context, f)
来变换 context。Entry
每一次资源调用都会创建一个
Entry
。Entry
包含了资源名、curNode(当前统计节点)、originNode(来源统计节点)等信息。
CtEntry
为普通的Entry
,在调用SphU.entry(xxx)
的时候创建。特性:Linked entry within current context(内部维护着parent
和child
)需要注意的一点:CtEntry 构造函数中会做调用链的变换,即将当前 Entry 接到传入 Context 的调用链路上(
setUpEntryFor
)。资源调用结束时需要
entry.exit()
。exit 操作会过一遍 slot chain exit,恢复调用栈,exit context 然后清空 entry 中的 context 防止重复调用。Node
Sentinel 里面的各种种类的统计节点:
StatisticNode
:最为基础的统计节点,包含秒级和分钟级两个滑动窗口结构。DefaultNode
:链路节点,用于统计调用链路上某个资源的数据,维持树状结构。ClusterNode
:簇点,用于统计每个资源全局的数据(不区分调用链路),以及存放该资源的按来源区分的调用数据(类型为StatisticNode
)。特别地,Constants.ENTRY_NODE
节点用于统计全局的入口资源数据。EntranceNode
:入口节点,特殊的链路节点,对应某个 Context 入口的所有调用数据。Constants.ROOT
节点也是入口节点。构建的时机:
EntranceNode
在ContextUtil.enter(xxx)
的时候就创建了,然后塞到 Context 里面。NodeSelectorSlot
:根据 context 创建DefaultNode
,然后 set curNode to context。ClusterBuilderSlot
:首先根据 resourceName 创建ClusterNode
,并且 set clusterNode to defaultNode;然后再根据 origin 创建来源节点(类型为StatisticNode
),并且 set originNode to curEntry。几种 Node 的维度(数目):
ClusterNode
的维度是 resourceDefaultNode
的维度是 resource * context,存在每个 NodeSelectorSlot 的map
里面EntranceNode
的维度是 context,存在 ContextUtil 类的contextNameNodeMap
里面- 来源节点(类型为
StatisticNode
)的维度是 resource * origin,存在每个 ClusterNode 的originCountMap
里面StatisticSlot
StatisticSlot
是 Sentinel 最为重要的类之一,用于根据规则判断结果进行相应的统计操作。entry 的时候:依次执行后面的判断 slot。每个 slot 触发流控的话会抛出异常(
BlockException
的子类)。若有BlockException
抛出,则记录 block 数据;若无异常抛出则算作可通过(pass),记录 pass 数据。exit 的时候:若无 error(无论是业务异常还是流控异常),记录 complete(success)以及 RT,线程数-1。
记录数据的维度:线程数+1、记录当前 DefaultNode 数据、记录对应的 originNode 数据(若存在 origin)、累计 IN 统计数据(若流量类型为 IN)。
各个插槽的功能:()
内为个人翻译
NodeSelectorSlot
(节点选择器插槽)
这个 slot 主要负责收集资源的路径,并将这些资源的调用路径以树状结构存储起来,用于根据调用路径进行流量控制。
//将参数`appA`(即**发起者**)注入到`enteance`(**上下文**)//创建上下文ContextUtil.enter("entrance1", "appA");//这里回应上文://>在每个资源调用的时候会自动创建一个Entry,而在我们实例化这个Entry的时候会启用插槽////这里选择了使用`SphU`这个API去调用 Entry nodeA = SphU.entry("nodeA");//验证该实体//实际开发时需要包裹异常`Block Exception` if (nodeA != null) { nodeA.exit(); }//结束 ContextUtil.exit();
上述代码通过 ContextUtil.enter()
创建了一个名为 entrance1
的上下文,同时指定调用发起者为 appA
;接着通过 SphU.entry()
请求一个 token,如果该方法顺利执行没有抛 BlockException
,表明 token 请求成功。
以上代码将在内存中生成以下结构:🎈(和可达性分析法的GCROOT是不是很相似)
machine-root / / EntranceNode1 / / DefaultNode(nodeA)
注意:每个 DefaultNode
由资源 ID 和输入名称来标识。换句话说,一个资源 ID 可以有多个不同入口的 DefaultNode。
ContextUtil.enter("entrance1", "appA"); Entry nodeA = SphU.entry("nodeA"); if (nodeA != null) { nodeA.exit(); } ContextUtil.exit(); ContextUtil.enter("entrance2", "appA"); nodeA = SphU.entry("nodeA"); if (nodeA != null) { nodeA.exit(); } ContextUtil.exit();
以上代码将在内存中生成以下结构:
machine-root / \ / \ EntranceNode1 EntranceNode2 / \ / \ DefaultNode(nodeA) DefaultNode(nodeA)
上面的结构可以通过调用 curl http://localhost:8719/tree?type=root
来显示:
EntranceNode: machine-root(t:0 pq:1 bq:0 tq:1 rt:0 prq:1 1mp:0 1mb:0 1mt:0)-EntranceNode1: Entrance1(t:0 pq:1 bq:0 tq:1 rt:0 prq:1 1mp:0 1mb:0 1mt:0)--nodeA(t:0 pq:1 bq:0 tq:1 rt:0 prq:1 1mp:0 1mb:0 1mt:0)-EntranceNode2: Entrance1(t:0 pq:1 bq:0 tq:1 rt:0 prq:1 1mp:0 1mb:0 1mt:0)--nodeA(t:0 pq:1 bq:0 tq:1 rt:0 prq:1 1mp:0 1mb:0 1mt:0)t:threadNum pq:passQps bq:blockedQps tq:totalQps rt:averageRt prq: passRequestQps 1mp:1m-passed 1mb:1m-blocked 1mt:1m-total
是不是很懵逼啊😵
😄:
解读一条:
结构对应树形,其中
EntranceNode
为根节点所以没有-,而后续代表子节点需要用-标识.🤨(有没有想起小米的自创题找一段树形代码的各个父节点啊)
参数理解:根据下面注解来:
很多就是
pass+block=total
这三个是指标哦后续在细分了平均和Request的Qps
1m是一分钟哦
具体解释:
- thread: 代表当前处理该资源的并发数;
- pass: 代表一秒内到来到的请求;
- blocked: 代表一秒内被流量控制的请求数量;
- success: 代表一秒内成功处理完的请求;
- total: 代表到一秒内到来的请求以及被阻止的请求总和;
- RT: 代表一秒内该资源的平均响应时间;
- 1m-pass: 则是一分钟内到来的请求;
- 1m-block: 则是一分钟内被阻止的请求;
- 1m-all: 则是一分钟内到来的请求和被阻止的请求的总和;
- exception: 则是一秒内业务本身异常的总和。
ClusterBuilderSlot
(构筑集群插槽)
此插槽用于构建资源的 ClusterNode
以及调用来源节点。ClusterNode
保持某个资源运行统计信息(响应时间、QPS、block 数目、线程数、异常数等)以及调用来源统计信息列表。调用来源的名称由 ContextUtil.enter(contextName,origin)
中的 origin
标记。可通过如下命令查看某个资源不同调用者的访问情况:curl http://localhost:8719/origin?id=caller
:
id: nodeAidx origin threadNum passedQps blockedQps totalQps aRt 1m-passed 1m-blocked 1m-total 1 caller1 0 0 0 0 0 0 0 0 2 caller2 0 0 0 0 0 0 0 0
一句话:统计做 参考类似
SELECT *
StatisticSlot
(静态资源插槽)
StatisticSlot
是 Sentinel 的核心功能插槽之一,用于统计实时的调用数据。
clusterNode
:资源唯一标识的 ClusterNode 的实时统计origin
:根据来自不同调用者的统计信息defaultnode
: 根据入口上下文区分的资源 ID 的 runtime 统计- 入口流量的统计
Sentinel 底层采用高性能的滑动窗口数据结构 LeapArray
来统计实时的秒级指标数据,可以很好地支撑写多于读的高并发场景。
滑动窗口!!重点🧑🏫回上面的图,那个均匀的环状有没有想到一致性hash算法啊
推荐文章:
Alibaba Sentinel LeapArray 源码分析 - 简书 (jianshu.com)
越少越重要
FlowSlot
这个 slot 主要根据预设的资源的统计信息,按照固定的次序,依次生效。如果一个资源对应两条或者多条流控规则,则会根据如下次序依次检验,直到全部通过或者有一个规则生效为止:
- 指定应用生效的规则,即针对调用方限流的;
- 调用方为 other 的规则;
- 调用方为 default 的规则。
DegradeSlot
这个 slot 主要针对资源的平均响应时间(RT)以及异常比率,来决定资源是否在接下来的时间被自动熔断掉。
SystemSlot
这个 slot 会根据对于当前系统的整体情况,对入口资源的调用进行动态调配。其原理是让入口的流量和当前系统的预计容量达到一个动态平衡。
注意系统规则只对入口流量起作用(调用类型为 EntryType.IN
),对出口流量无效。可通过 SphU.entry(res, entryType)
指定调用类型,如果不指定,默认是EntryType.OUT
。
引入依赖:
<?xml version="1.0" encoding="UTF-8"?><project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.3.2.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <groupId>com.hariji</groupId> <artifactId>Sentinel</artifactId> <version>0.0.1-SNAPSHOT</version> <name>Sentinel</name> <description>Sentinel</description> <properties> <java.version>1.8</java.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> <version>2.3.2.RELEASE</version> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>1.2.78</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.3.2.RELEASE</version> <type>pom</type> <scope>import</scope> </dependency> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-dependencies</artifactId> <!--suppress UnresolvedMavenProperty --> <version>Hoxton.SR9</version> <type>pom</type> <scope>import</scope> </dependency> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-alibaba-dependencies</artifactId> <version>2.2.6.RELEASE</version> <type>pom</type> <scope>import</scope> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- 引入 Spring Cloud Alibaba Nacos Discovery 相关依赖,将 Nacos 作为注册中心,并实现对其的自动配置 --> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId> <version>2.2.6.RELEASE</version> </dependency> <!-- https://mvnrepository.com/artifact/com.alibaba.cloud/spring-cloud-starter-alibaba-sentinel --> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-sentinel</artifactId> <version>2.2.6.RELEASE</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId> <version>2.2.6.RELEASE</version> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build></project>
编写配置类:
spring: application: name: demo-provider cloud: # Sentinel 配置项,对应 SentinelProperties 配置属性类 sentinel: enabled: true # 是否开启。默认为 true 开启 eager: true # 是否饥饿加载。默认为 false 关闭 transport: dashboard: 127.0.0.1:7070 # Sentinel 控制台地址 filter: url-patterns: /** # 拦截请求的地址。默认为 /*
配置详细解释
Sentinel 配置项,以spring.cloud.sentinel
开头,对应 SentinelProperties 配置属性类。①
enabled
配置项,设置是否开启 Sentinel,默认为true
开启,所以一般不用主动设置。如果关闭 Sentinel 的功能,例如说在本地开发的时候,可以设置为false
关闭。②
eager
配置项,设置是否饥饿加载,默认为false
关闭。默认情况下,Sentinel 是延迟初始化,在首次使用到 Sentinel 才进行初始化。通过设置为true
时,在项目启动时就会将 Sentinel 直接初始化,完成向 Sentinel 控制台进行注册。③
transport.dashboard
配置项,设置 Sentinel 控制台地址。④
filter.url-patterns
配置项,设置拦截请求的地址,默认为/*
。在 Sentinel 的子项目
sentinel-spring-webmvc-adapter
中,对 SpringMVC 进行适配,通过 SentinelWebInterceptor 拦截器,实现对 SpringMVC 的请求的拦截,使用 Sentinel 进行保护。通过filter.url-patterns
配置项,可以定义该拦截器的拦截请求地址。不过要注意,因为
filter.url-patterns
配置项的默认值为/*
,只能拦截根目录的请求,显然不满足我们的日常需求,因此修改成了/**
拦截所有请求。可以阅读下《SpringMVC Ant 路径匹配》文章。
BlockException 处理器
先来对 BlockException 异常做个简单的了解,在被 Sentinel block 的时候,就会抛出它。BlockException 是一个异常抽象基类,其有 5 个实现类,刚好对应 Sentinel 的 5 种流量控制手段,如下图所示:
旁白君:暂时找不到 block 适合翻译成什么单词,相对最贴切的可能是阻塞…
在 SentinelWebInterceptor 拦截器中,当请求满足配置的 Sentinel block 的条件时,Sentinel 会抛出 BlockException 异常。通过定义 BlockExceptionHandler 接口的实现类,可以实现对 BlockException 的异常处理。
默认情况下,BlockExceptionHandler 有一个默认的 DefaultBlockExceptionHandler 实现类,返回 Block 字符串提示。代码如下:
public class DefaultBlockExceptionHandler implements BlockExceptionHandler { @Override public void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception { // ... 省略其它代码 PrintWriter out = response.getWriter(); out.print("Blocked by Sentinel (flow limiting)"); }}
显然,在我们使用 SpringMVC 提供 Restful API 时,直接返回字符串提示是不合适的,因为一般是返回 JSON 字符串,例如说:
{ "code": 1024, "msg": "Blocked by Sentinel (flow limiting)"}
因此,我们自定义的 CustomBlockExceptionHandler 实现类,直接抛出 BlockException 异常,最终交给自定义的 SpringMVC 全局异常处理器 ,将 BlockException 异常处理成 JSON 字符串提示返回。代码如下:
// CustomBlockExceptionHandler.java@Componentpublic class CustomBlockExceptionHandler implements BlockExceptionHandler { @Override public void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception { throw e; }}
友情提示:如果胖友对 SpringMVC 的全局异常处理器不了解的话,可以看看 《芋道 Spring Boot SpringMVC 入门》文章的 「5. 全局异常处理」小节。
事前工作:
搭建Sentinel
参考
命令
java -Dserver.port=8080 -Dcsp.sentinel.dashboard.server=localhost:8080 -Dproject.name=sentinel-dashboard -Dsentinel.dashboard.auth.username=sentinel -Dsentinel.dashboard.auth.password=sentinel-jar sentinel-dashboard-1.8.2.jar
出错:
jar包下载出现错误
解决方法:
1.构筑源码打包并执行2.对于jar包用新的命令
1.直接编译运行
Java -jar sentinel-dashboard-1.8.2.jar --server.port=7070
2.通过加入引号来界定配置java ‘-Dserver.port=7070’ ‘-Dcsp.sentinel.dashboard.server=localhost:7070’ ‘-Dproject.name=sentinel-dashboard’ -jar sentinel-dashboard-1.8.2.jar
按照文档编写代码测试结果如下:
打开控制台:
后续步骤:
② 使用浏览器,访问下 http://127.0.0.1:7070/ 地址,进入 Sentinel 控制台。此时,我们可以看到 demo-provider
应用。如下图所示:
③ 使用浏览器,访问下 http://127.0.0.1:8080/demo/echo 接口 10 次。然后点击 Sentinel 控制台的「实时监控」菜单,可以看到该接口的请求情况。如下图所示:
④ 点击 Sentinel 控制台的「簇点链路」菜单,可以看到 /demo/echo
资源。如下图所示:
⑤ 点击 /demo/echo
资源所在列的「流控」按钮,弹出「新增流控规则」。填写流控规则,如下图所示:
- 这里,我们创建的是比较简单的规则,仅允许
/demo/echo
资源被每秒调用一次。 - 更多详细的配置项的说明,胖友后续一定要认真看《Sentinel 官方文档 —— 流量控制》文章,这是 Sentinel 提供的多种规则中最最最常用的一种。
⑥ 点击「新增」按钮,完成流控规则的添加。此时,会自动跳转到「流控规则」菜单。如下图所示:
⑦ 使用浏览器,访问 http://127.0.0.1:8080/demo/echo 接口两次,会有一次被 Sentinel 流量控制而拒绝,最终返回如下 JSON 字符串:
{ "msg": "请求被拦截,拦截类型为 FlowException", "code": 1024}
- 流量控制对应 FlowException 异常,因此这里会看到哈。
此时,点击 Sentinel 控制台的「实时监控」菜单,可以看到该接口被拒绝的统计。如下图所示:
Sentinel特性:(多环境和兼容性)
以官方文档为准:
1.如何使用(基于在 WebServlet 环境中使用)
引入依赖
<dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-sentinel</artifactId></dependency>
简单测试例子:
@SpringBootApplicationpublic class Application { public static void main(String[] args) { SpringApplication.run(ServiceApplication.class, args); }}@RestControllerpublic class TestController { @GetMapping(value = "/hello") @SentinelResource("hello") @ sentinelresource 注释用于标识一个资源是速率有限还是降级。在上面的示例中,注释的‘ hello’属性指向资源名称。 @ sentinelresource 还提供了诸如 blockHandler、 blockHandlerClass 和 fallback 等属性来识别速率限制或降级操作。有关详细信息,请参阅哨兵注释支持。 public String hello() { return "Hello Sentinel"; }}
2.仪表盘
这一部分在上述有启动仪表盘
基于SpringCloud配置:
application.yaml
spring: cloud: sentinel: transport: port: 8719 dashboard: localhost:8080
在 spring.cloud.Sentinel.transport.port
中指定的端口号将在应用程序的相应服务器上启动一个 HTTP 服务器,该服务器将与 Sentinel 仪表板进行交互。例如,如果在 Sentinel 指示板中添加了速率限制规则,规则数据将被推送到 HTTP 服务器并接收,HTTP 服务器再将规则注册到 Sentinel。
3.支持OpenFeign(稍后再议)
需要提前理解
Open Feign
的作用.以及优势:
Ribbon
Ribbon 是 Netflix开源的基于HTTP和TCP等协议负载均衡组件Ribbon 可以用来做客户端负载均衡,调用注册中心的服务
Ribbon的使用需要代码里手动调用目标服务,请参考官方示例:https://github.com/Netflix/ribbon
Feign
Feign是Spring Cloud组件中的一个轻量级RESTful的HTTP服务客户端Feign内置了Ribbon,用来做客户端负载均衡,去调用服务注册中心的服务。
Feign的使用方式是:使用Feign的注解定义接口,调用这个接口,就可以调用服务注册中心的服务
Feign支持的注解和用法请参考官方文档:https://github.com/OpenFeign/feign
Feign本身不支持Spring MVC的注解,它有一套自己的注解
OpenFeign
OpenFeign是Spring Cloud 在Feign的基础上支持了Spring MVC的注解,如@RequesMapping等等。
OpenFeign的@FeignClient可以解析SpringMVC的@RequestMapping注解下的接口,
并通过动态代理的方式产生实现类,实现类中做负载均衡并调用其他服务。// user-api 子项目public interface SysUserResource { @GetMapping("/test") Object getUser();}// user-client 子项目 , 依赖了user-api 子项目// 其他业务模块可以直接依赖此模块,通过调用接口即可完成服务的远程调用,open-feign会对此类做动态代理// name = "user-center" 是被调用的服务实例名称@FeignClient(name = "user-center")public interface SysUserResourceClient extends SysUserResource {}// user-impl 子项目@RestControllerpublic class SysUserResourceImpl implements SysUserResource { @Override Object getUser(){ // do something }}// role-impl 子项目 , 依赖了 user-client 子项目@RestControllerpublic class SysRoleResourceImpl implements SysRoleResource { @Resource private SysUserResource sysUserResource; @OverrideObject test(){ sysUserResource.getUser();}
需要注意,@RequesMapping不能在类名上与@FeignClient同时使用
5.4. RestTemplate Support
Spring Cloud Alibaba Sentinel supports the protection of RestTemplate
service calls using Sentinel. To do this, you need to add the @SentinelRestTemplate
annotation when constructing the RestTemplate
bean.
Spring Cloud alibaba Sentinel 支持使用 Sentinel 保护 RestTemplate 服务调用。为此,在构造 RestTemplate bean 时需要添加@sentinelresttemplate 注释。
@Bean@SentinelRestTemplate(blockHandler = "handleException", blockHandlerClass = ExceptionUtil.class)public RestTemplate restTemplate() { return new RestTemplate();}
The attribute of the @SentinelRestTemplate
annotation support flow control(blockHandler
, blockHandlerClass
) and circuit breaking(fallback
, fallbackClass
).
@ sentinelresttemplate 注释支持流控制(blockHandler、 blockHandlerClass)和断路(fallback、 fallbackClass)的属性。
==
The blockHandler
or fallback
is the static method of blockHandlerClass
or fallbackClass
.
blockHandler 或 fallback 是 blockHandlerClass 或 fallbackClass 的静态方法。
The parameter and return value of method in @SentinelRestTemplate
is same as org.springframework.http.client.ClientHttpRequestInterceptor#interceptor
, but it has one more parameter BlockException
to catch the exception by Sentinel.
@ sentinelresttemplate 中方法的参数和返回值与 org.springframework.http.client 相同。ClientHttpRequestInterceptor # interceptor,但是它还有一个参数 BlockException 来捕获 Sentinel 的异常。
The method signature of handleException
in ExceptionUtil
above should be like this:
在 exceptionutill 中 handleException 的方法签名应该是这样的:
public class ExceptionUtil { public static ClientHttpResponse handleException(HttpRequest request, byte[] body, ClientHttpRequestExecution execution, BlockException exception) { ... }}
When the application starts, it will check if the @SentinelRestTemplate annotation corresponding to the flow control or circuit breaking method exists, if it does not exist, it will throw an exception. 当应用程序启动时,它将检查对应于流控制或断路方法的@sentinelresttemplate 注释是否存在,如果不存在,它将抛出异常。 |
|
---|---|
The attribute of the @SentinelRestTemplate
annotation is optional.
@ sentinelresttemplate 注释的属性是可选的。
It will return RestTemplate request block by sentinel
when you using RestTemplate
blocked by Sentinel. You can override it by your own logic. We provide SentinelClientHttpResponse
to handle the response.
当您使用由 Sentinel 阻塞的 RestTemplate 时,它将按照 Sentinel 返回 RestTemplate 请求块。你可以用自己的逻辑推翻它。我们提供 SentinelClientHttpResponse 来处理响应。
Sentinel RestTemplate provides two granularities for resource rate limiting:
Sentinel RestTemplate 为资源速率限制提供两种粒度:
httpmethod:schema://host:port/path
: Protocol, host, port and path端口/路径: Protocol,host,port and path
httpmethod:schema://host:port
: Protocol, host and port端口: Protocol,host and port
Take Http GET https://www.taobao.com/test as an example. The corresponding resource names have two levels of granularities, GET:https://www.taobao.com and GET:https://www.taobao.com/test . 以 Http GET https://www.taobao.com/test 为例,对应的资源名称有两个级别的粒度,GET: https://www.taobao.com 和 GET: https://www.taobao.com/test。 |
|
---|---|
5. 动态数据源支持
SentinelProperties 提供 datasource 属性来配置数据源。
例如,配置4个数据源:
spring.cloud.sentinel.datasource.ds1.file.file=classpath: degraderule.jsonspring.cloud.sentinel.datasource.ds1.file.rule-type=flow#spring.cloud.sentinel.datasource.ds1.file.file=classpath: flowrule.json#spring.cloud.sentinel.datasource.ds1.file.data-type=custom#spring.cloud.sentinel.datasource.ds1.file.converter-class=JsonFlowRuleListConverter#spring.cloud.sentinel.datasource.ds1.file.rule-type=flowspring.cloud.sentinel.datasource.ds2.nacos.server-addr=localhost:8848spring.cloud.sentinel.datasource.ds2.nacos.data-id=sentinelspring.cloud.sentinel.datasource.ds2.nacos.group-id=DEFAULT_GROUPspring.cloud.sentinel.datasource.ds2.nacos.data-type=jsonspring.cloud.sentinel.datasource.ds2.nacos.rule-type=degradespring.cloud.sentinel.datasource.ds3.zk.path = /Sentinel-Demo/SYSTEM-CODE-DEMO-FLOWspring.cloud.sentinel.datasource.ds3.zk.server-addr = localhost:2181spring.cloud.sentinel.datasource.ds3.zk.rule-type=authorityspring.cloud.sentinel.datasource.ds4.apollo.namespace-name = applicationspring.cloud.sentinel.datasource.ds4.apollo.flow-rules-key = sentinelspring.cloud.sentinel.datasource.ds4.apollo.default-flow-rule-value = testspring.cloud.sentinel.datasource.ds4.apollo.rule-type=param-flow
该方法遵循 Spring Cloud Stream Binder 的配置,`TreeMap
用于内部存储,比较器为 String.CASE _ insensitive _ order
。
D1、 ds2、 ds3、 ds4是
` ReadableDataSource的名称,可以根据需要进行编码。文件,
zk,
`nacos,
apollo` 指的是特定的数据源。下面的配置分别是这些数据源的特定配置。
每个数据源有3个常见的配置项: data-type
, converter-class
and rule-type
.
数据类型是指转换器。`Spring Cloud alibaba Sentinel
默认提供两个嵌入值: json 和 xml (如果没有指定,默认值是 json)。如果不想使用嵌入的 json 或 xml Converter,还可以填写自定义,以指示您将定义自己的 Converter,然后配置 Converter-class。您需要为此配置指定类的完整路径。
Rule-type 指的是数据源中的规则类型(flow、 degrad、 authority、 system、 param-flow、 gw-flow、 gw-api-group)。
默认情况下不支持 XML 格式。要使其有效,需要添加
jackson-dataformat-XML
依赖项。
Sentinel上手
半原理半开发
1.流量控制
流量控制 · alibaba/Sentinel Wiki · GitHub
目的: 保障应用的高可用性
原理:监控应用流量的 QPS 或并发线程数等指标,当达到指定的阈值时对流量进行控制,以避免被瞬时的流量高峰冲垮
🚩:指标想到工作原理中的插槽,那么多的插槽指标如何工作呢?
FlowSlot
会根据预设的规则,结合前面 NodeSelectorSlot
、ClusterBuilderSlot
、StatisticSlot
统计出来的实时信息进行流量控制。
同一个资源可以创建 。FlowSlot
会对该资源的所有限流规则依次遍历,直到有规则触发限流或者所有规则遍历完毕。
一条限流规则主要由下面几个因素组成,我们可以组合这些元素来实现不同的限流效果:
resource
:资源名,即限流规则的作用对象count
: 限流阈值grade
: 限流阈值类型(QPS 或并发线程数)limitApp
: 流控针对的调用来源,若为default
则不区分调用来源strategy
: 调用关系限流策略controlBehavior
: 流量控制效果(直接拒绝、Warm Up、匀速排队)