微服务负载均衡器Ribbon

1.什么是Ribbon
目前主流的负载方案分为以下两种:
        集中式负载均衡,在消费者和服务提供方中间使用独立的代理方式进行负载,有硬件的(比如 F5),也有软件的(比如 Nginx)。
        客户端根据自己的请求情况做负载均衡,Ribbon 就属于客户端自己做负载均衡。
        Spring Cloud Ribbon是基于Netflix Ribbon 实现的一套 客户端的负载均衡工具, Ribbon客户端组件提供一系列的完善的配置,如超 时,重试等。通过 Load Balancer 获取到服务提供的所有机器实例,Ribbon会自动基于某种规则(轮询,随机)去调用这些服务。Ribbon也 可以实现我们自己的负载均衡算法。
1.1 客户端的负载均衡
        例如spring cloud中的ribbon,客户端会有一个服务器地址列表,在发送请求前通过负载均衡算法选择一个服务器,然后进行访问,这是 客户端负载均衡;即在客户端就进行负载均衡算法分配。
1.2 服务端的负载均衡
        例如Nginx,通过Nginx进行负载均衡,先发送请求,然后通过负载均衡算法,在多个服务器之间选择一个进行访问;即在服务器端再进 行负载均衡算法分配。
1.3 常见负载均衡算法
        随机,通过随机选择服务进行执行,一般这种方式使用较少;
        轮训,负载均衡默认实现方式,请求来之后排队处理;
        加权轮训,通过对服务器性能的分型,给高配置,低负载的服务器分配更高的权重,均衡各个服务器的压力;
        地址Hash,通过客户端请求的地址的HASH值取模映射进行服务器调度。 ip --->hash
        最小链接数,即使请求均衡了,压力不一定会均衡,最小连接数法就是根据服务器的情况,比如请求积压数等参数,将请求分配到当前压力最小的服务器上。 最小活跃数
2. Nacos使用Ribbon
nacos-discovery依赖了ribbon,可以不用再引入ribbon依赖
2) 添加 @LoadBalanced 注解
@Configuration
 public class RestConfig {
 @Bean
 @LoadBalanced
 public RestTemplate restTemplate() {
 return new RestTemplate();
 }
 }
3) 修改controller
@Autowired
 private RestTemplate restTemplate;

 @RequestMapping(value = "/findOrderByUserId/{id}")
 public R findOrderByUserId(@PathVariable("id") Integer id) {
 // RestTemplate调用
 //String url = "http://localhost:8020/order/findOrderByUserId/"+id;
 //模拟ribbon实现
 //String url = getUri("mall‐order")+"/order/findOrderByUserId/"+id;
// 添加@LoadBalanced
String url = "http://mall‐order/order/findOrderByUserId/"+id;
R result = restTemplate.getForObject(url,R.class);

return result;
}
3 Ribbon负载均衡策略
IRule
这是所有负载均衡策略的父接口, 里边的核心方法就是 choose 方法,用来选择一个服务实例
AbstractLoadBalancerRule
AbstractLoadBalancerRule 是一个抽象类,里边主要定义了一个 ILoadBalancer ,这里定义它的目的主要是 辅助负责均衡策略选取合适的服务端实 例。
RandomRule
看名字就知道,这种负载均衡策略就是 随机选择一个服务实例 ,看源码我们知道,在 RandomRule 的无参构造方法中初始化了一个 Random 对象, 然后在它重写的choose 方法又调用了 choose(ILoadBalancer lb, Object key) 这个重载的 choose 方法,在这个重载的 choose 方法中,每次利用 random对象生成一个不大于服务实例总数的随机数,并将该数作为下标所以获取一个服务实例。
RoundRobinRule
RoundRobinRule 这种负载均衡策略叫做线性 轮询负载均衡策略 。这个类的 choose(ILoadBalancer lb, Object key) 函数整体逻辑是这样的:开启 一个计数器count ,在 while 循环中遍历服务清单,获取清单之前先通过 incrementAndGetModulo 方法获取一个下标,这个下标是一个不断自增长 的数先加1 然后和服务清单总数取模之后获取到的(所以这个下标从来不会越界),拿着下标再去服务清单列表中取服务,每次循环计数器都会加
1 ,如果连续 10 次都没有取到服务,则会报一个警告 No available alive servers after 10 tries from load balancer: XXXX
RetryRule (在轮询的基础上进行重试)
看名字就知道这种负载均衡策略带有 重试 功能。首先 RetryRule 中又定义了一个 subRule ,它的实现类是 RoundRobinRule ,然后在 RetryRule
choose(ILoadBalancer lb, Object key) 方法中,每次还是采用 RoundRobinRule 中的 choose 规则来选择一个服务实例,如果选到的实例正常就返
回,如果选择的服务实例为 null 或者已经失效,则在失效时间 deadline 之前不断的进行重试(重试时获取服务的策略还是 RoundRobinRule 中定义的
策略),如果超过了 deadline 还是没取到则会返回一个 null
WeightedResponseTimeRule 权重 —nacos 的NacosRule ,Nacos还扩展了一个自己的基于配置的权重扩展 WeightedResponseTimeRule是 RoundRobinRule 的一个子类,在 WeightedResponseTimeRule 中对 RoundRobinRule 的功能进行了扩展,
WeightedResponseTimeRule 中会根据每一个实例的运行情况来给计算出该实例的一个 权重 ,然后在挑选实例的时候则根据权重进行挑选,这样能 够实现更优的实例调用。WeightedResponseTimeRule 中有一个名叫 DynamicServerWeightTask 的定时任务,默认情况下每隔 30 秒会计算一次 各个服务实例的权重,权重的计算规则也很简单, 如果一个服务的平均响应时间越短则权重越大,那么该服务实例被选中执行任务的概率也就越大
ClientConfigEnabledRoundRobinRule
ClientConfigEnabledRoundRobinRule 选择策略的实现很简单,内部定义了 RoundRobinRule choose 方法还是采用了 RoundRobinRule 的 choose方法,所以它的选择策略 RoundRobinRule 的选择策略一致 ,不赘述。
BestAvailableRule
BestAvailableRule 继承自 ClientConfigEnabledRoundRobinRule ,它在 ClientConfigEnabledRoundRobinRule 的基础上主要增加了根据 loadBalancerStats中保存的服务实例的状态信息来 过滤掉失效的服务实例的功能,然后顺便找出并发请求最小的服务实例来使用。 然而 loadBalancerStats有可能为 null ,如果 loadBalancerStats null ,则 BestAvailableRule 将采用它的父类即 ClientConfigEnabledRoundRobinRule的服务选取策略(线性轮询)。
ZoneAvoidanceRule
默认规则 ,复合判断server所在区域的性能和server的可用性选择服务器。
ZoneAvoidanceRule PredicateBasedRule 的一个实现类,只不过这里多一个过滤条件, ZoneAvoidanceRule 中的过滤条件是以
ZoneAvoidancePredicate 为主过滤条件和以
AvailabilityPredicate 为次过滤条件组成的一个叫做 CompositePredicate 的组合过滤条件,过滤成功之后,继续采用线性轮询
( RoundRobinRule ) 的方式从过滤结果中选择一个出来。 AvailabilityFilteringRule (先过滤掉故障实例,再选择并发较小的实例) 过滤掉一直连接失败的被标记为circuit tripped的后端Server,并过滤掉那些高并发的后端Server或者使用一个AvailabilityPredicate来 包含过滤server的逻辑,其实就是检查status里记录的各个Server的运行状态。
3.2.1 修改默认负载均衡策略
@Configuration
 public class RibbonConfig {

 /**
 * 全局配置
 * 指定负载均衡策略
 * @return
 */
 @Bean
public IRule iRule() {
 // 指定使用Nacos提供的负载均衡策略(优先调用同一集群的实例,基于随机权重)
return new NacosRule();
}
}
注意:此处有坑。 不能写在@SpringbootApplication注解的@CompentScan扫描得到的地方,否则自定义的配置类就会被所有的 RibbonClients共享。 不建议这么使用,推荐yml方式
利用@RibbonClient指定微服务及其负载均衡策略。
 @SpringBootApplication(exclude = {DataSourceAutoConfiguration.class,
 DruidDataSourceAutoConfigure.class})
 //@RibbonClient(name = "mall‐order",configuration = RibbonConfig.class)
 //配置多个 RibbonConfig不能被@SpringbootApplication的@CompentScan扫描到,否则就是全局配置的效果
 @RibbonClients(value = {
 // 在SpringBoot主程序扫描的包外定义配置类
 @RibbonClient(name = "mall‐order",configuration = RibbonConfig.class),
 @RibbonClient(name = "mall‐account",configuration = RibbonConfig.class)
 })
public class MallUserRibbonDemoApplication {

 public static void main(String[] args) {
 SpringApplication.run(MallUserRibbonDemoApplication.class, args);
 }
 }
配置文件 :调用指定微服务提供的服务时,使用对应的负载均衡算法
修改application.yml
 # 被调用的微服务名
 mall‐order:
 ribbon:
 # 指定使用Nacos提供的负载均衡策略(优先调用同一集群的实例,基于随机&权重)
 NFLoadBalancerRuleClassName: com.alibaba.cloud.nacos.ribbon.NacosRule
3.2.2 自定义负载均衡策略
通过实现 IRule 接口可以自定义负载策略,主要的选择服务逻辑在 choose 方法中。
1) 实现基于Nacos权重的负载均衡策略
@Slf4j
 public class NacosRandomWithWeightRule extends AbstractLoadBalancerRule {

 @Autowired
 private NacosDiscoveryProperties nacosDiscoveryProperties;

 @Override
 public Server choose(Object key) {
 DynamicServerListLoadBalancer loadBalancer = (DynamicServerListLoadBalancer) getLoadBalancer();
 String serviceName = loadBalancer.getName();
 NamingService namingService = nacosDiscoveryProperties.namingServiceInstance();
 try {
 //nacos基于权重的算法
 Instance instance = namingService.selectOneHealthyInstance(serviceName);
 return new NacosServer(instance);
 } catch (NacosException e) {
 log.error("获取服务实例异常:{}", e.getMessage());
 e.printStackTrace();
 }
 return null;
 }
 @Override
 public void initWithNiwsConfig(IClientConfig clientConfig) {

 }
 }
2) 配置自定义的策略
2.1)配置文件:
修改application.yml
# 被调用的微服务名
 mall‐order:
 ribbon:
 # 自定义的负载均衡策略(基于随机&权重)
 NFLoadBalancerRuleClassName: com.tuling.mall.ribbondemo.rule.NacosRandomWithWeightRule
3.3 饥饿加载
在进行服务调用的时候,如果网络情况不好,第一次调用会超时。
Ribbon默认懒加载,意味着只有在发起调用的时候才会创建客户端。
开启饥饿加载,解决第一次调用慢的问
 ribbon:
 eager‐load:
 # 开启ribbon饥饿加载
 enabled: true
 # 配置mall‐user使用ribbon饥饿加载,多个使用逗号分隔
 clients: mall‐order
源码对应属性配置类:RibbonEagerLoadProperties
测试:

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

CentOS 7 使用异步网络框架Libevent

CentOS 7 安装Libevent库 libevent github地址:https://github.com/libevent/libevent 步骤1:首先,你需要下载libevent的源代码。你可以从github或者源代码官方网站下载。并上传至/usr/local/source_code/ 步骤2:下载完成后&…

Springboot整合MybatisPlus及分页功能

1 引入pom <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot</artifactId><version>2.7.14</version> </dependency> <dependency><groupId>com.baomidou</groupId><a…

【考研数据结构代码题8】三元组表示的稀疏矩阵快速转置

题目&#xff1a;三元组表示的稀疏矩阵快速转置 考点&#xff1a;矩阵的压缩与存储 难度&#xff1a;★★★ 稀疏矩阵压缩存储 三元组结构 //三元组结构 typedef struct {int row;int col;int value; } Term; 核心代码 // 转置函数&#xff0c;a为原矩阵&#xff0c;b存放转…

AIGC专题报告:生成式人工智能用例汇编

今天分享的是AIGC系列深度研究报告&#xff1a;《AIGC专题报告&#xff1a;生成式人工智能用例汇编》。 &#xff08;报告出品方&#xff1a;德勤&#xff09; 报告共计&#xff1a;16页 生成式人工智能&#xff08;AI&#xff09;的兴起 生成式AI给人类文明创造了无限的可…

STM32 CAN通信自定义数据包多帧连发乱序问题

场景&#xff1a; can标准帧中每一帧只能传输8字节&#xff0c;而应用中传输一包的内容往往超过8字节&#xff0c;因此需要把一个包拆成多个帧发送&#xff0c;接收端才把收到的多帧重新组装成一个完整的包 问题描述 在一问一答的两块板间通信&#xff0c;多帧连发是能够按照…

致远M3 反序列化RCE漏洞复现(XVE-2023-24878)

0x01 产品简介 M3移动办公是致远互联打造的一站式智能工作平台&#xff0c;提供全方位的企业移动业务管理&#xff0c;致力于构建以人为中心的智能化移动应用场景&#xff0c;促进人员工作积极性和创造力&#xff0c;提升企业效率和效能&#xff0c;是为企业量身定制的移动智慧…

基于51单片机音乐盒设计( proteus仿真+程序+原理图+PCB+报告+讲解视频)

音乐盒 主要功能&#xff1a;仿真原理图PCB图程序设计&#xff1a;设计报告实物图资料清单&#xff08;提供资料清单所有文件&#xff09;&#xff1a;资料下载链接&#xff1a; 基于51单片机音乐盒仿真设计( proteus仿真程序原理图PCB报告讲解视频&#xff09; 仿真图proteus …

【LeetCode刷题】--67.二进制求和

67.二进制求和 方法&#xff1a;模拟计算 class Solution {public String addBinary(String a, String b) {StringBuilder ans new StringBuilder();int carry 0;for(int ia.length()-1,jb.length()-1;i>0||j>0;i--,j--){int sum carry;sum i >0 ? a.charAt(i) …

web:[WUSTCTF2020]朴实无华

题目 点开页面显示如下 页面显示了一行报错&#xff1a;Cannot modify header information - headers already sent by (output started at /var/www/html/index.php:3) in /var/www/html/index.php on line 4 意思为不能修改报头信息-报头已经发送(输出开始于/var/www/html/i…

深度学习之基于Pytorch照片图像转漫画风格网络系统

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 以下是一个基本的设计介绍&#xff1a; 数据准备&#xff1a;收集足够的真实照片和漫画图像&#xff0c;用于训练模…

富士康转移产线和中国手机海外设厂,中国手机出口减少超5亿部

富士康和苹果转移生产线对中国手机制造造成了巨大的影响&#xff0c;除此之外&#xff0c;中国手机企业纷纷在海外设厂也在减少中国手机的出口&#xff0c;2022年中国的手机出口较高峰期减少了5.2亿部。 手机是中国的大宗出口商品&#xff0c;不过公开的数据显示2022年中国的手…

ElementUI table+dialog实现一个简单的可编辑的表格

table组件如何实现可编辑呢&#xff1f; 我的需求是把table组件那样的表格&#xff0c;实现它点击可以弹出一个框&#xff0c;然后在这个框里面输入你的东西&#xff0c;然后将他回显回去&#xff0c;当然&#xff0c;输入的有可能是时间啥的。 为什么要弹出弹层不在框上直接…

缓存雪崩、击穿、穿透及解决方案_保证缓存和数据库一致性

文章目录 缓存雪崩、击穿、穿透1.缓存雪崩造成缓存雪崩解决缓存雪崩 2. 缓存击穿造成缓存击穿解决缓存击穿 3.缓存穿透造成缓存穿透解决缓存穿透 更新数据时&#xff0c;如何保证数据库和缓存的一致性&#xff1f;1. 先更新数据库&#xff1f;先更新缓存&#xff1f;解决方案 2…

C语言函数练习(超基础超详细)

ps:题目来源于pta平台。 1. int sum(int m, int n) {int sum0;for(int im; i<n; i){sumi;}return sum; } 2. int max(int a, int b) {if(a>b)return a;else return b; } 3. double dist( double x1, double y1, double x2, double y2 ) {return sqrt((x1-x2)*(x1…

【Flink】Standalone运行模式

独立模式是独立运行的&#xff0c;不依赖任何外部的资源管理平台&#xff1b;当然独立也是有代价的&#xff1a;如果资源不足&#xff0c;或者出现故障&#xff0c;没有自动扩展或重分配资源的保证&#xff0c;必须手动处理。所以独立模式一般只用在开发测试或作业非常少的场景…

Docker+ Jenkins+Maven+git自动化部署

环境&#xff1a;Centos7 JDK1.8 Maven3.3.9 Git 2.40 Docker 20.10.17 准备工作&#xff1a; 安装Docker Centos7默认的yum安装的docker是1.13&#xff0c;版本太低&#xff0c;很多镜像都要Docker版本要求&#xff0c;升级Docker版本。 卸载已安装Docker: yum …

NeurIPS 2023 | RGIB:对抗双边图噪声的鲁棒图学习

▐ 摘要 链接预测[1,2]是图学习的一种基础任务&#xff0c;用于判断图中的两个节点是否可能相连&#xff0c;被广泛应用于药物发现、知识图谱补全和在线问答等实际场景。尽管图神经网络&#xff08;Graph Neural Network&#xff0c;GNN&#xff09;在该问题的性能上取得了显著…

我做了一个世界杯数据可视化网站······

感兴趣的小伙伴可以进去看看&#xff1a;主页https://messimeimei.github.io/world-cup-visualization.github.io/&#xff0c;可能会比较卡 经过2个月的工作&#xff0c;我完成了80%的工作量&#xff0c;并成功将静态网站进行了部署。并对页面进行了更新。不过当前虽然完成了…

ESP32之避障

ESP32之避障 图片 程序 int Led27;//定义LED 接口 int buttonpin4; //定义光遮断传感器接口 int val;//定义数字变量val void setup() { pinMode(Led,OUTPUT);//定义LED 为输出接口 pinMode(buttonpin,INPUT);//定义避障传感器为输出接口 } void loop() {Serial.begin(9600);…

性能相关的闪存特性

一、多Plane操作 上章提到若干个Plane组成Die或者叫LUN,即一个Die上有多个Plane 每次进行写操作时&#xff0c;控制器先将数据写入页缓存中&#xff0c;等同一个Die上另一个Plane也写数据的时候&#xff0c;再同时写入&#xff0c;原来单独操作一个Plane的时间变成了可以同时做…