Spring Cloud-Ribbon
Spring Cloud-Ribbon
简介
Spring Cloud Ribbon是基于NetFlix Ribbon实现的一套客户端 均衡负载工具
Ribbon = 负载均衡 + RestTemplate
简单的说,Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负载均衡算法和服务调用。Ribbon客户端组件提供一系列完善的配置项如连接超时,重试等。简单的说,就是在配置文件中列出Load Balancer(简称LB)后面的所有机器,Ribbon会自动的帮助你基于某种规则(如简单轮询,随机连接等)去连接这些机器。我们很容易使用Ribbon实现自定义的负载均衡算法。
注意:Ribbon已经进入了维护阶段,Spring Cloud意向使用
LoadBalancer
替换
LB负载均衡(Load Balance)是什么
简单的说就是将用户的请求平摊的分配到多个服务上,从而达到HA(高可用)。
常见的负载均衡右软件Nginx
、LVS
、硬件 F5等。
集中式LB
即在服务的消费方和提供方之间使用独立的LB设施(可以是硬件,如F5,也可以是软件,如Nginx),由该设施负责把访问请求通过某种策略转发至服务的提供方。
进程内LB
将LB逻辑集成到消费方,消费方从服务注册中心获知有哪些地址可用,然后自己再从这些地址中选择出一个合适的服务器。
Ribbon就属于进程内LB,它只是一个类库,集成于消费方进程,消费方通过它来获取到服务提供方的地址。
Ribbon本地负载均衡客户端 VS Nginx服务端负载均衡区别
Nginx是服务器负载均衡,客户端所有请求都会将给nginx,然后由nginx实现转发请求。即负载均衡是由服务端实现的。(被动 消费方被动)
Ribbon本地负载均衡,在调用微服务接口时候,会在注册中心上获取注册信息服务列表之后缓存到JVM本地,从而在本地实现RPC远程调用技术。(主动 消费方主动)
总结
Ribbon其实就是一个软负载均衡的客户端软件,它可以和其他所需请求的客户端结合使用,和Eureka结合只是其中的一个实例。
负载均衡流程
- 发起请求,请求发送到了服务提供者的负载均衡拦截器上。
- 通过Ribbon的负载均衡客户端通过服务名称向注册中心请求服务列表。
- 获取到服务列表后,通过设定的负载均衡策略去选出一个服务,获取服务信息。
- 将获取到的服务信息替换原来的请求中的服务名称。
- 发送真正有意义的http请求。
Pom依赖
<!-- https://mvnrepository.com/artifact/org.springframework.cloud/spring-cloud-starter-netflix-ribbon --> |
在之前的入门项目Eureka案例中我们并没有显式的添加Ribbon依赖,但是依旧可以使用负载均衡的原因就是我们引入的spring-cloud-starter-netflix-eureka-client
中自带了Ribbon,consul,zookeeper也是如此(新版本的Spring Cloud中所有原本自带Ribbon的包都被换为了Spring Cloud自己的Load Balancer)
RestTemplate
Ribbon核心组件IRule
public interface IRule{ |
Ribbon默认自带的负载策略
Ribbon的负载均衡鬼册是一个叫做IRule的接口来定义,每一个子接口都是一种规则。
IRule:根据特定算法中从服务列表中选取一个要访问的服务。
类名 | 策略 |
---|---|
com.netflix.loadbalancer.RoundRobinRule |
轮询 |
com.netflix.loadbalancer.RandomRule |
随机 |
com.netflix.loadbalancer.RetryRule |
先按照RoundRobinRule的策略获取服务,如果获取服务失败则在指定时间内会进行重试,获取可用的服务。 |
WeightResponseTimeRule |
对RoundRobinRule的扩展,响应速度越快的实例选择权重越大,越容易被选择。 |
BestAvailableRule |
会先过滤掉由于多次访问故障而处于断路器跳闸状态的服务,然后选择一个并发量最小的服务。 |
AvailabilityFilteringRule |
先过滤掉故障实例,再选择并发较小的实例 |
ZoneAvoidanceRule |
默认规则,复合判断Server所在区域的性能和server的可用性选择服务器 |
如何替换负载均衡策略
修改工程Cloud-eureka-consumer-order80
修改Ribbon负载均衡策略有两种配置方式,配置类方式和YAML方式。
方式一 创建配置类
编写配置类替换的负载均衡策略。配置负载均衡策略,就是将对应的IRule的Bean添加到 Spring 容器中。
MySelfRule
|
该方式会针对所有该消费者调用的服务进行策略修改。
也可以一通过在主启动类上添加@RibbonClient注解为服务指定负载均衡策略。
//指定那个服务使用那种配置类中的策略 |
修改主启动类
|
方式二 YAML
配置文件,为指定服务配置负载均衡策略。前缀为服务名称。
CLOUD-PAYMENT-SERVICE: |
这里我们就为cloud-payment-service
服务配置了随机的负载均衡规则。
启动测试 http://localhost:80/consumer/payment/get/1547134537463402497
{ |
{ |
连续访问两次都是8001服务响应说明已经不再是轮询机制 而是随机。
Ribbon默认负载均衡算法–轮询原理
负载均衡算法:rest接口第几次请求数%服务器集群总数量 = 实际调用服务器位置下标,每次服务重启后rest接口计数从1开始。
源码
package com.netflix.loadbalancer; |
手写轮询算法
前提:注释掉主启动类上的@RibbonClient
和RestTemplate配置类中的@LoadBalancer
旨在取消Ribbon的负载均衡。
在
Cloud-eureka-provider-payment8001
Cloud-eureka-provider-payment8002
新增接口
public String getPaymentLB(){
return serverPort;
}新建接口
loadBalancer
public interface LoadBalancer {
ServiceInstance instances(List<ServiceInstance> serviceInstances); //获取服务器实例 实现负载均衡算法
}实现接口 实现自己的轮询算法
MyLB
(根据Ribbon轮询源码)package com.dyw.springcloud.lb;
import org.springframework.cloud.client.ServiceInstance;
import org.springframework.stereotype.Component;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
/**
* @author Devil
* @since 2022-07-24-19:57
*/
public class MyLB implements LoadBalancer{
//初始化计数器
private AtomicInteger atomicInteger = new AtomicInteger(0);
//负载均衡轮询策略计算出的节点索引
public final int getAndIncrement(){
//当前节点
int current;
//下一个节点
int next;
do { //自旋锁
current = this.atomicInteger.get();
//防止Integer类型变量越界
next = current >= Integer.MAX_VALUE ? 0 : current+1;
}while (!this.atomicInteger.compareAndSet(current,next));//AtomicInteger从内存偏移两种取出数据 如果取出的数据与current一样就将内存中的数据改为next并且返回true 是为了保障线程安全而使用一种乐观锁
System.out.println("*****next"+next);
return next;
}
public ServiceInstance instances(List<ServiceInstance> serviceInstances) {
//通过负载均衡算法计算出的索引取 服务器列表取出对应索引的服务器实例并返回
int index = getAndIncrement() % serviceInstances.size();
return serviceInstances.get(index);
}
}控制器类
public class OrderController {
// private static final String PAYMENT_URL = "http://localhost:8001";
private static final String PAYMENT_URL = "http://CLOUD-PAYMENT-SERVICE";
RestTemplate restTemplate;
//这个则是我们自定义的负载均衡的接口
private LoadBalancer loadBalancer;
//必须有它才能获取注册中心中的服务器实例列表
private DiscoveryClient discoveryClient;
public String getPaymentLB(){
//首先获取指定服务的服务器实例列表
List<ServiceInstance> instances = discoveryClient.getInstances("CLOUD-PAYMENT-SERVICE");
//判断服务器列表是否为空 如果是直接返回空
if (instances==null||instances.size()<=0){
return null;
}
//通过自定义的负载均衡算法获取服务实例
ServiceInstance serviceInstance = loadBalancer.instances(instances);
//再获取服务器的实例的uri 通过这个uri借助resttemplate去访问服务 这也是我们去掉RestTemplate配置类上@LoadBalancer的原因 加上这个注解会让RestTemplate通过服务名称去访问服务 而我们使用的是uri 是ip端口的格式 服务器列表中是没有这样的服务的所以会报错
URI uri = serviceInstance.getUri();
return restTemplate.getForObject(uri+"/payment/lb", String.class);
}
}启动测试
8001
8002
8001
8002轮询成功,算法成功。
饥饿加载
Ribbon默认是采用懒加载,即第一次访问时才回去创建LoadBalanceClient,请求时间会很长。
而饥饿加载则会在项目启动时创建,降低第一次访问的好事,通过下面配置开启饥饿加载:
ribbon: |
总结
- Ribbon负载均衡规则
- 规则接口时IRule
- 默认实现时ZoneAvoidanceRule,根据zone选择服务列表,然后轮询
- 负载均衡配置方式
- 代码方式:配置灵活,但修改时需要重新打包发布
- 配置方式:直观,方便,无序重新打包发布,但无法做全局配置。
- 饥饿加载
- 开启饥饿加载
- 指定饥饿加载的微服务名称