【Spring Cloud】Ribbon详细介绍及底层原理分析

目录

核心功能

Ribbon 的工作原理

底层原理及代码详解

1. ServerList

2. IRule

3. IPing

4. ServerListUpdater

使用场景

实际使用示例

1. 使用 RestTemplate 与 Ribbon 集成

2. 使用 Feign 与 Ribbon 集成


Ribbon 是 Netflix 开源的一款客户端负载均衡器,它可以在客户端实现负载均衡,并与服务发现机制(如 Eureka)集成,为客户端提供动态的服务实例选择。

Spring Cloud 对 Ribbon 进行了封装,使其更容易集成到 Spring Boot 和 Spring Cloud 应用中。

核心功能

  1. 客户端负载均衡:在客户端实现负载均衡逻辑,选择适合的服务实例进行请求。
  2. 服务发现集成:与 Eureka 等服务发现组件集成,实现动态服务发现。
  3. 多种负载均衡策略:提供多种负载均衡策略,如轮询、随机、响应时间加权等。
  4. 故障检测与恢复:通过 Ping 机制检测实例健康状态。

Ribbon 的工作原理

Ribbon 的工作原理可以分为以下几个主要步骤:

  1. 服务实例列表获取

    • Ribbon 通过 ServerList 接口获取服务实例列表。这个列表可以来自 Eureka 等服务发现组件。
  2. 负载均衡策略

    • Ribbon 通过 IRule 接口选择负载均衡策略。默认策略是 RoundRobinRule(轮询),但可以配置其他策略。
  3. 健康检查

    • Ribbon 通过 IPing 接口进行健康检查,确保请求只发送到健康的实例。默认实现是 NoOpPing,不做任何健康检查。
  4. 请求转发

    • Ribbon 客户端根据选定的负载均衡策略和健康检查结果,从实例列表中选择一个实例并转发请求。

底层原理及代码详解

Ribbon 的核心组件包括 ServerList, IRule, IPing, ServerListUpdater 等。下面对这些核心组件进行详细介绍和代码解析:

1. ServerList

ServerList 接口负责获取服务实例列表。它有两个主要方法:

  • List<T> getInitialListOfServers(): 获取初始的服务实例列表。
  • List<T> getUpdatedListOfServers(): 获取更新后的服务实例列表。

Spring Cloud Ribbon 中,默认使用 DiscoveryEnabledNIWSServerList 从 Eureka 获取服务实例。

代码示例

public interface ServerList<T extends Server> {List<T> getInitialListOfServers();List<T> getUpdatedListOfServers();
}
2. IRule

IRule 接口定义了负载均衡策略。它有一个主要方法:

  • Server choose(Object key): 根据负载均衡策略选择一个服务实例。

常用的实现类有 RoundRobinRule(轮询策略), RandomRule(随机策略), WeightedResponseTimeRule(加权响应时间策略)等。

代码示例

public interface IRule {Server choose(Object key);void setLoadBalancer(ILoadBalancer lb);ILoadBalancer getLoadBalancer();
}

RoundRobinRule 实现

public class RoundRobinRule extends AbstractLoadBalancerRule {private AtomicInteger nextServerCyclicCounter;@Overridepublic Server choose(Object key) {ILoadBalancer lb = getLoadBalancer();return choose(lb, key);}private Server choose(ILoadBalancer lb, Object key) {if (lb == null) {return null;}Server server = null;int count = 0;while (server == null && count++ < 10) {List<Server> reachableServers = lb.getReachableServers();List<Server> allServers = lb.getAllServers();int upCount = reachableServers.size();int serverCount = allServers.size();if ((upCount == 0) || (serverCount == 0)) {return null;}int nextServerIndex = incrementAndGetModulo(serverCount);server = allServers.get(nextServerIndex);if (server == null) {Thread.yield();continue;}if (server.isAlive() && (server.isReadyToServe())) {return server;}server = null;}return server;}private int incrementAndGetModulo(int modulo) {for (;;) {int current = nextServerCyclicCounter.get();int next = (current + 1) % modulo;if (nextServerCyclicCounter.compareAndSet(current, next)) {return next;}}}
}
3. IPing

IPing 接口定义了服务实例的健康检查机制。它有一个主要方法:

  • boolean isAlive(Server server): 检查服务实例是否健康。

常用的实现类有 NoOpPing(不进行健康检查), PingUrl(通过 URL 进行健康检查)等。

代码示例

public interface IPing {boolean isAlive(Server server);
}

PingUrl 实现

public class PingUrl implements IPing {private String pingAppendString = "/health";@Overridepublic boolean isAlive(Server server) {String urlStr = "http://" + server.getHost() + ":" + server.getPort() + pingAppendString;HttpURLConnection conn = null;try {URL url = new URL(urlStr);conn = (HttpURLConnection) url.openConnection();conn.setConnectTimeout(2000);conn.setReadTimeout(2000);conn.setRequestMethod("GET");int code = conn.getResponseCode();return (code == 200);} catch (Exception e) {return false;} finally {if (conn != null) {conn.disconnect();}}}
}
4. ServerListUpdater

ServerListUpdater 接口定义了更新服务实例列表的机制。Spring Cloud Ribbon 使用 PollingServerListUpdater 定期从 Eureka 获取最新的服务实例列表。

代码示例

public interface ServerListUpdater {void start(UpdateAction updateAction);void stop();interface UpdateAction {void doUpdate();}
}

PollingServerListUpdater 实现

public class PollingServerListUpdater implements ServerListUpdater {private final ScheduledExecutorService refreshExecutor;private final AtomicBoolean isActive = new AtomicBoolean(false);private final int refreshIntervalMs;public PollingServerListUpdater() {this(10000);}public PollingServerListUpdater(int refreshIntervalMs) {this.refreshIntervalMs = refreshIntervalMs;this.refreshExecutor = Executors.newScheduledThreadPool(1);}@Overridepublic void start(UpdateAction updateAction) {if (isActive.compareAndSet(false, true)) {refreshExecutor.scheduleWithFixedDelay(() -> {try {updateAction.doUpdate();} catch (Exception e) {// Log error}}, 0, refreshIntervalMs, TimeUnit.MILLISECONDS);}}@Overridepublic void stop() {if (isActive.compareAndSet(true, false)) {refreshExecutor.shutdown();}}
}

使用场景

  1. 服务注册与发现

    • 与 Eureka 或其他服务发现组件结合使用,实现服务注册与动态发现。
  2. 客户端负载均衡

    • 在客户端实现负载均衡,分散请求压力,提高系统的可靠性和可用性。
  3. 多种负载均衡策略应用

    • 根据实际需求选择不同的负载均衡策略,如轮询、随机、加权响应时间等,以优化服务调用性能。
  4. 健康检查与故障恢复

    • 通过健康检查机制,确保请求只发送到健康的服务实例,提升系统的稳定性。

实际使用示例

1. 使用 RestTemplate 与 Ribbon 集成

配置 RestTemplate

@Configuration
public class RibbonConfig {@Bean@LoadBalancedpublic RestTemplate restTemplate() {return new RestTemplate();}
}

使用 RestTemplate 调用服务

@RestController
public class MyController {@Autowiredprivate RestTemplate restTemplate;@GetMapping("/call")public String callService() {return restTemplate.getForObject("http://service-a/hello", String.class);}
}
2. 使用 Feign 与 Ribbon 集成

Feign Client 定义

@FeignClient(name = "service-a")
public interface MyFeignClient {@GetMapping("/hello")String sayHello();
}

使用 Feign Client

@RestController
public class MyController {@Autowiredprivate MyFeignClient myFeignClient;@GetMapping("/call")public String callService() {return myFeignClient.sayHello();}
}

Ribbon 是微服务架构中实现客户端负载均衡的重要工具,通过它可以显著提升服务调用的效率和可靠性。

在深入学习 Ribbon 的过程中,我们从基本概念和原理出发,了解了其核心组件及工作机制,并通过实际代码示例掌握了如何在 Spring Cloud 中集成和配置 Ribbon。

希望通过本文的介绍,能够帮助你更好地理解和掌握 Ribbon,为你的微服务架构实践提供有力支持。

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

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

相关文章

算法训练营day04

一、24. 两两交换链表中的节点 题目链接&#xff1a;https://leetcode.cn/problems/swap-nodes-in-pairs/description/ 文章讲解&#xff1a;https://programmercarl.com/0024.%E4%B8%A4%E4%B8%A4%E4%BA%A4%E6%8D%A2%E9%93%BE%E8%A1%A8%E4%B8%AD%E7%9A%84%E8%8A%82%E7%82%B9.h…

国产操作系统上Vim的详解03--使用Vundle插件管理器来安装和使用插件 _ 统信 _ 麒麟 _ 中科方德

原文链接&#xff1a;国产操作系统上Vim的详解03–使用Vundle插件管理器来安装和使用插件 | 统信 | 麒麟 | 中科方德 Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇在国产操作系统上使用Vundle插件管理器来安装和使用Vim插件的详解文章。Vundle是Vim的一款强大的插…

【数据可视化系列】使用Python和Seaborn绘制相关性热力图

热力图&#xff08;Heatmap&#xff09;是一种数据可视化工具&#xff0c;它通过使用颜色的深浅来展示数据矩阵中数值的大小或密度。在热力图中&#xff0c;每种颜色的深浅代表数据的一个特定值或值的范围&#xff0c;通常使用红色、黄色和绿色等颜色渐变来表示数据的热度&…

frida timed out

从Android Q(10)开始&#xff0c;Google引入了一种新的机制&#xff0c;加快了app的启动时间 Android USAP 进程启动流程 adb shell su ps -A | grep usaproot 9917 1032 6577052 13676 __skb_wait_for_more_packets 0 S usap64 root 9928 1032 6577052…

1、项目介绍:为什么要做此项目。

项目介绍&#xff1a;为什么要做此项目。 全栈开发博客实战项目&#xff1a;前后端开发流程以及项目部署 随着互联网的蓬勃发展&#xff0c;全栈开发成为了越来越受欢迎的趋势。前端开发和后端开发之间的紧密合作和协同工作已经成为了现代软件开发中的重要组成部分。然而&…

【TB作品】MSP430F5529 单片机,数字时钟设计与实现,整点时通过蜂鸣器播放音乐进行报时

基于单片机的数字时钟设计与实现 作品名称 基于MSP430单片机的OLED显示数字时钟 作品功能 本作品实现了一个具有时间显示和整点报时功能的数字时钟。通过OLED屏幕显示当前时间&#xff0c;用户可以通过按键设置时间&#xff0c;并在整点时通过蜂鸣器播放音乐进行报时。 作…

Emacs Verilog mode 简单使用指南

Emacs 是一个强大的文本编辑器,以其高度可定制和扩展能力著称。在硬件描述语言(HDL)的开发中,Verilog 是一种广泛使用的语言,而 Emacs 的 Verilog mode 为 Verilog 开发提供了强有力的支持。本文将详细介绍如何使用 Emacs Verilog mode 来高效编写 Verilog 代码,帮助读者…

MySQL JDBC编程

前言&#x1f440;~ 上一章我们介绍了数据库的索引与事务&#xff0c;本章节我们介绍JDBC编程&#xff0c;使用java代码和数据库打交道 如果各位对文章的内容感兴趣的话&#xff0c;请点点小赞&#xff0c;关注一手不迷路&#xff0c;如果内容有什么问题的话&#xff0c;欢迎各…

双指针算法题笔记

1、移动零 class Solution {public void moveZeroes(int[] nums) {int left0;int right0;for(right0;right<nums.length;right){if(nums[right]!0){if(nums[left]0){int tempnums[left];nums[left]nums[right];nums[right]temp;}left;}}} } 两个指针将一个数组划分三个部分&…

使用OpenPCDet训练与测试多传感器融合模型BEVFusion,OPenPCdet代码架构介绍

引言 在自动驾驶领域&#xff0c;多传感器融合技术是一种常见的方法&#xff0c;用于提高感知系统的准确性和鲁棒性。其中&#xff0c;BevFusion是一种流行的融合方法&#xff0c;可以将来自不同传感器的数据进行融合&#xff0c;生成具有丰富信息的鸟瞰图&#xff08;BEV&…

探索800G数据中心的高速布线解决方案

随着技术的快速进步&#xff0c;数据中心正以前所未有的速度迅速发展。虽然100G和400G数据中心已经普及&#xff0c;但800G数据中心正逐渐流行并展现出增长趋势。由于对高速数据传输的需求呈指数级增长&#xff0c;因此需要高效、可靠的线缆连接解决方案。本文将介绍800G数据中…

Cell-在十字花科植物中年生和多次开花多年生开花行为的互相转化-文献精读21

Reciprocal conversion between annual and polycarpic perennial flowering behavior in the Brassicaceae 在十字花科植物中年生和多次开花多年生开花行为的互相转化 亮点 喜马拉雅须弥芥 和 内华达糖芥 是两个多年生植物模型 MADS-box 基因的剂量效应决定了一年生、二年生…

[图解]企业应用架构模式2024新译本讲解11-领域模型4

1 00:00:00,160 --> 00:00:01,870 好&#xff0c;到这里的话 2 00:00:02,620 --> 00:00:05,060 文字处理器的产品对象就生成了 3 00:00:06,880 --> 00:00:09,180 同样下面就是电子表格 4 00:00:10,490 --> 00:00:11,480 电子表格也同样的 5 00:00:11,490 -->…

字符输出流

在Java的IO流中专门提供了用于字符输出的流对象 PrintWriter 。该对象具有自动换行刷新缓冲字符输出流。特点是可以按行写出字符串&#xff0c;并且可通过 println(); 方法实现自动换行。 public class TestPrintWriter {public static void main(String[] args){//创建字符输…

如何轻松修改Windows远程连接的端口号

为了增强远程连接的安全性&#xff0c;最好修改默认的远程桌面协议&#xff08;RDP&#xff09;端口号。以下步骤将指导您如何修改Windows注册表中的端口设置&#xff0c;并相应地更新防火墙规则。 一、修改注册表中的端口号 打开注册表编辑器&#xff1a; 按下Win R键&#…

用python编撰一个电脑清理程序

自制一个电脑清理程序&#xff0c;有啥用呢&#xff1f;在电脑不装有清理软件的时候&#xff0c;可以解决自己电脑内存不足的情况。 1、设想需要删除指定文件夹中的临时文件和缓存文件。以下是代码。 import os import shutil def clean_folder(folder_path): for root,…

11 IP协议 - IP协议头部

什么是 IP 协议 IP&#xff08;Internet Protocol&#xff09;是一种网络通信协议&#xff0c;它是互联网的核心协议之一&#xff0c;负责在计算机网络中路由数据包&#xff0c;使数据能够在不同设备之间进行有效的传输。IP协议的主要作用包括寻址、分组、路由和转发数据包&am…

计算机网络 期末复习(谢希仁版本)第4章

路由器&#xff1a;查找转发表&#xff0c;转发分组。 IP网的意义&#xff1a;当互联网上的主机进行通信时&#xff0c;就好像在一个网络上通信一样&#xff0c;看不见互连的各具体的网络异构细节。如果在这种覆盖全球的 IP 网的上层使用 TCP 协议&#xff0c;那么就…

【六袆 - Java】Java 驱动连接Oracle数据库; Java单元测试 连接Oracle;

Java 驱动连接Oracle数据库 JDK8 Oracle驱动包 ORACLE8 测试用例代码 @Testpublic void oracleConn(){ResultSet rs = null;Statement stmt = null;Connection conn = null

股票数据集1-纳斯达克NASDAQ 100简介

数据集信息 纳斯达克 100 指数 由纳斯达克证券交易所 上市的 100 家最大的非金融公司发行的股票证券组成&#xff0c;根据公司市值设置不同权重的加权指数&#xff0c;权重还受某些规则影响如公司影响力。该指数仅限于来自单一交易所的公司&#xff0c;并且没有任何金融公司。 …