自动化配置

由于 Ribbon 中定义的每一个接口都有多种不同的策略实现,同时这些接口之间又有一定的依赖关系,Spring Cloud Ribbon 中的自动化配置能够很方便的自动化构建接口的具体实现,接口如下:

  • IClientConfig:Ribbon 的客户端配置,默认采用 com.netflix.client.config.DefaultClientConfigImpl 实现。
  • IRule:Ribbon 的负载均衡策略,默认采用 com.netflix.loadbalancer.ZoneAvoidanceRule 实现,该策略能够在多区域环境下选择最佳区域的实例进行访问
  • IPing:Ribbon 的实例检查策略,默认采用 com.netflix.loadbalancer.NoOpPing 实现,该检查策略是一个特殊的实现,实际上他并不会检查实例是否可用,而是始终返回 true ,默认认为所有服务实例都是可以使用
  • ServerList<Server>:服务实例清单的维护机制,默认采用 com.netflix.loadbalancer.ConfigurationBasedServerList 实现。
  • ServerListFilter<Server>:服务实例清单过滤机制,默认采用 org.springframework.cloud.netflix.ribbon.ZonePreferenceServerListFilter 实现,该策略能够优先过滤出与请求调用方处理同区域的服务实现
  • ILoadBalancer:负载均衡器,默认采用 com.netflix.loadbalancer.ZoneAwareLoadBalancer 实现,他具备了区域感知的能力

这些自动化配置内容是在没有引入
Spring
Cloud
Eureka
等服务治理框架时如此,在同时引入
Eureka

Ribbon
依赖时,自动化配置会有一些不同。通过自动化配置的实现,我们可以轻松的实现客户端的负载均衡,同时,针对一些个性化的需求,我们也可以方便的替换上面的这些默认实现,示例代码:

@Configurable

public class RibbonConfiguration {

@Bean

public IRule ribbonRule(IClientConfig clientConfig) {

return new
WeightedResponseTimeRule();

}

}

可以使用 @RibbonClient
注解来实现更细粒度的客户端配置,创建 ConsumerHelloserviceApplication 主类,使用 @RibbonClient 注解来指定服务使用 WebServiceRibbonConfiguration 配置,代码如下:

@RibbonClient (name = "ORG.DRSOFT.WEBSERVICE.HELLOSERVICE", configuration = RibbonConfiguration.class)

@EnableDiscoveryClient

@SpringBootApplication

public class ConsumerHelloserviceApplication {

public static
void
main(String[] args) {

SpringApplication.run(ConsumerHelloserviceApplication.class, args);

}

@LoadBalanced

@Bean

public RestTemplate createRestTemplate() {

return new
RestTemplate();

}

}

注意:如果需要使服务单独配置,那么配置类不能放在@ComponentScan所扫描的当前包下以及子包下,否则我们自定义的这个配置类就会被所有的Ribbon客户端所共享,@SpringBootApplication 注解中就包含了 @ComponentScan 注解,因此必须使配置类处于不同的包以及子包。

如果需要设置默认配置或者多个@RibbonClient
注解,可以使用 @RibbonClients 注解,其 defaultConfiguration
可以设置默认的自定义配置,配置类不限定所属包,value
可以设置多个 @RibbonClient 注解,示例代码如下:

@RibbonClients (defaultConfiguration = RibbonConfiguration.class)

@EnableDiscoveryClient

@SpringBootApplication

public class ConsumerHelloserviceApplication {

public static
void
main(String[] args) {

SpringApplication.run(ConsumerHelloserviceApplication.class, args);

}

@Bean

@LoadBalanced

public RestTemplate restTemplate() {

return new
RestTemplate();

}

}


Eureka
结合

当在
Spring
Cloud
的应用中同时应用
Spring
Cloud Ribbon 和 Spring Cloud Eureka 依赖时,会触发 Eureka 中实现的对 Ribbon 的自动化配置,接口实现如下:

  • ServerList<Server>:服务实例清单的维护机制,默认采用 com.netflix.niws.loadbalancer.DiscoveryEnabledNIWSServerList实现,该实现会将服务清单列表交给 Eureka 的服务治理机制来进行维护
  • IPing:Ribbon 的实例检查策略,默认采用 com.netflix.niws.loadbalancer.NIWSDiscoveryPing 实现,该实现将实例检查的任务交给了服务治理框架来进行维护

负载均衡策略

IRule
接口其Ribbon
实现了非常多的选择策略,下面我们来详细说明一下IRule 接口的定义如下:

public interface IRule{

public Server choose(Object key);

public
void
setLoadBalancer(ILoadBalancer lb);

public ILoadBalancer getLoadBalancer();

}

IRule
接口的具体实现:

  • AbstractLoadBalancerRule:负载均衡策略的抽象类,在该抽象类中定义了负载均衡器 ILoadBalancer 对象,该对象能够根据在具体实现选择服务策略时,获取到一些负载均衡器中维护的信息来作为分配依据,并以此设计一些算法来实习针对特定场景的高效策略
  • RandomRule:该策略实现了从服务实例清单中随机选择一个服务实例的功能,查看源码可以看到 choose 函数实现,使用 Random类的 nextInt 函数来随机获取一个随机数,并将随机数作为座位 upList 的索引值来返回具体实例

    public Server choose(ILoadBalancer lb, Object key) {

    if (lb == null) {

    return null;

    }

    Server server = null;

    while (server == null) {

    if (Thread.interrupted()) {

    return null;

    }

    List<Server> upList = lb.getReachableServers();

    List<Server> allList = lb.getAllServers();

    int serverCount = allList.size();

    if (serverCount == 0) {

    return null;

    }

    int index = rand.nextInt(serverCount);

    server = upList.get(index);

    if (server == null) {

    Thread.yield();

    continue;

    }

    if (server.isAlive()) {

    return (server);

    }

    server = null;

    Thread.yield();

    }

    return server;

    }

  • RoundRobinRule:该策略实现了按照线性轮询的方式依次选择每个服务实例的功能,其详细结构和
    RandomRule类似,具体实现如下:

    public Server choose(ILoadBalancer lb, Object key) {

    if (lb == null) {

    log.warn("no load balancer");

    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)) {

    log.warn("No up servers available from load balancer: " + lb);

    return null;

    }

    int nextServerIndex = incrementAndGetModulo(serverCount);

    server = allServers.get(nextServerIndex);

    if (server == null) {

    /* Transient. */

    Thread.yield();

    continue;

    }

    if (server.isAlive() && (server.isReadyToServe())) {

    return (server);

    }

    // Next.

    server = null;

    }

    if (count >= 10) {

    log.warn("No available alive servers after 10 tries from load balancer: " + lb);

    }

    return server;

    }

  • WeightedResponseTimeRule:该策略是对
    RoundRobinRule的扩展,增加了根据实例的运行情况来计算权重,并根据权重来挑选实例,以达到更优的分配效果,实现主要有三个核心内容:
    • 定时任务:在初始化的时候会通过 serverWeightTimer.schedule(new DynamicServerWeightTask(), 0, serverWeightTaskTimerInterval); 启动一个定时任务,用来为每个服务实例计算权重,该任务默认30秒执行一次。
    • 权重计算:维护实例的权重计算过程通过
      maintainWeights 函数实现,具体代码如下:

      public
      void
      maintainWeights() {

    ILoadBalancer lb = getLoadBalancer();

    if (lb == null) {

    return;

    }

    if (!serverWeightAssignmentInProgress.compareAndSet(false, true)) {

    return;

    }

    try {

    logger.info("Weight adjusting job started");

    AbstractLoadBalancer nlb = (AbstractLoadBalancer) lb;

    LoadBalancerStats stats = nlb.getLoadBalancerStats();

    if (stats == null) {

    return;

    }

    double totalResponseTime = 0;

    for (Server server : nlb.getAllServers()) {

    ServerStats ss = stats.getSingleServerStat(server);

    totalResponseTime += ss.getResponseTimeAvg();

    }

    Double weightSoFar = 0.0;

    List<Double> finalWeights = new ArrayList<Double>();

    for (Server server : nlb.getAllServers()) {

    ServerStats ss = stats.getSingleServerStat(server);

    double weight = totalResponseTime - ss.getResponseTimeAvg();

    weightSoFar += weight;

    finalWeights.add(weightSoFar);

    }

    setWeights(finalWeights);

    } catch (Exception e) {

    logger.error("Error calculating server weights", e);

    } finally {

    serverWeightAssignmentInProgress.set(false);

    }

    }

    计算规则为
    weightSoFar + totalResponseTime - 实例的平均响应时间,其中
    weightSoFar
    初始化为零,并且每计算好一个权重值需要累加到
    weightSoFar
    上供下一次计算使用,假设有4个实例A、B、C、D,他们的平均响应时间为10、40、80、100,所以总响应时间是 230,每个实例的权重为总响应时间与实例自身的平均响应时间的差的累积所得,所以A、B、C、D的权重分别如下所示:

    • 实例A:230-10=220
    • 实例B:220+(230-40)=410
    • 实例C:410+(230-80)=560
    • 实例D:560+(230-100)=690

    需要注意的是,这里的权重值只表示各实例权重区间的上限,每个实例的权重区间如下:

    • 实例A:[0,220]
    • 实例B:[220,410]
    • 实例C:[410,560]
    • 实例D:[560,690]

    权重区间的宽度越大,而权重区间的宽度越大,被选中的概率越高

    • 实例选择:生成一个[0,最大权重值]区间内的随机数,然后遍历权重列表的索引值去服务实例列表中获取具体的实例。
  • ClientConfigEnabledRoundRobinRule:该策略比较特殊,他本身并没有实现什么特殊的处理策略逻辑,在内部定义了一个RoundRobinRule 策略,我们不会直接使用该策略,但可以通过继承该策略,默认的choose实现了线性轮询策略,在子类中做一些高级策略时通常有可能存在一些无法实施的情况,那么就可以用父类的实现作为备选,后面的高级策略都是基于该类的扩展
  • BestAvailableRule:该策略继承自
    ClientConfigEnabledRoundRobinRule
    ,在实现中他注入了负载均衡器的统计对象
    LoadBalancerStats,同时在具体的
    choose
    方法中利用
    LoadBalancerStats
    保存的实例统计信息来选择满足要求的实例,通过负载均衡器中维护的所有服务实例,会过滤掉故障的实例,并找出并发请求最小的一个,所以该策略的特性是可选出最空闲的实例,代码如下:

    public Server choose(Object key) {

    if (loadBalancerStats == null) {

    return super.choose(key);

    }

    List<Server> serverList = getLoadBalancer().getAllServers();

    int minimalConcurrentConnections = Integer.MAX_VALUE;

    long currentTime = System.currentTimeMillis();

    Server chosen = null;

    for (Server server: serverList) {

    ServerStats serverStats = loadBalancerStats.getSingleServerStat(server);

    if (!serverStats.isCircuitBreakerTripped(currentTime)) {

    int concurrentConnections = serverStats.getActiveRequestsCount(currentTime);

    if (concurrentConnections < minimalConcurrentConnections) {

    minimalConcurrentConnections = concurrentConnections;

    chosen = server;

    }

    }

    }

    if (chosen == null) {

    return super.choose(key);

    } else {

    return chosen;

    }

    }

笔记:Spring Cloud Ribbon 客户端配置详解的更多相关文章

  1. Spring Cloud Eureka 常用配置详解,建议收藏!

    前几天,栈长分享了 <Spring Cloud Eureka 注册中心集群搭建,Greenwich 最新版!>,今天来分享下 Spring Cloud Eureka 常用的一些参数配置及说 ...

  2. Spring学习 6- Spring MVC (Spring MVC原理及配置详解)

    百度的面试官问:Web容器,Servlet容器,SpringMVC容器的区别: 我还写了个文章,说明web容器与servlet容器的联系,参考:servlet单实例多线程模式 这个文章有web容器与s ...

  3. Spring 入门 web.xml配置详解

    Spring 入门 web.xml配置详解 https://www.cnblogs.com/cczz_11/p/4363314.html https://blog.csdn.net/hellolove ...

  4. 笔记:Spring Cloud Ribbon 客户端负载均衡

    Spring Cloud Ribbon 是一个基于 HTTP 和 TCP 的客户端负载均衡工具,基于 Netflix Ribbon 实现,通过Spring Cloud 的封装,可以让我们轻松的将面向服 ...

  5. 跟我学Spring Cloud(Finchley版)-20-Spring Cloud Config-Git仓库配置详解 原

    在跟我学Spring Cloud(Finchley版)-19-配置中心-Spring Cloud Config 一节中,已实现使用Git仓库作为Config Server的后端存储,本节详细探讨如何配 ...

  6. Spring Cloud Zuul 限流详解(附源码)(转)

    在高并发的应用中,限流往往是一个绕不开的话题.本文详细探讨在Spring Cloud中如何实现限流. 在 Zuul 上实现限流是个不错的选择,只需要编写一个过滤器就可以了,关键在于如何实现限流的算法. ...

  7. Spring Cloud(十二):Spring Cloud Zuul 限流详解(附源码)(转)

    前面已经介绍了很多zuul的功能,本篇继续介绍它的另一大功能.在高并发的应用中,限流往往是一个绕不开的话题.本文详细探讨在Spring Cloud中如何实现限流. 在 Zuul 上实现限流是个不错的选 ...

  8. Spring Cloud Ribbon客户端负载均衡(四)

    序言 Ribbon 是一个客户端负载均衡器(Nginx 为服务端负载均衡),它赋予了应用一些支配 HTTP 与 TCP 行为的能力,可以得知,这里的客户端负载均衡也是进程内负载均衡的一种.它在 Spr ...

  9. Spring Cloud Ribbon 客户端负载均衡 4.3

      在分布式架构中,服务器端负载均衡通常是由Nginx实现分发请求的,而客户端的同一个实例部署在多个应用上时,也需要实现负载均衡.那么Spring Cloud中是否提供了这种负载均衡的功能呢?答案是肯 ...

随机推荐

  1. sqoop:mysql和Hbase/Hive/Hdfs之间相互导入数据

    1.安装sqoop 请参考http://www.cnblogs.com/Richardzhu/p/3322635.html 增加了SQOOP_HOME相关环境变量:source ~/.bashrc  ...

  2. react-dom.js 源码

    /** *以下这是 react-dom.js 的源代码 * ReactDOM v15.3.1 * * Copyright 2013-present, Facebook, Inc. * All righ ...

  3. 暴力破解MD5的实现(MapReduce编程)

    本文主要介绍MapReduce编程模型的原理和基于Hadoop的MD5暴力破解思路. 一.MapReduce的基本原理 Hadoop作为一个分布式架构的实现方案,它的核心思想包括以下几个方面:HDFS ...

  4. Java多线程,判断其他线程是否结束的方法

    方法1:通过Thread类中的isAlive()方法判断线程是否处于活动状态. 线程启动后,只要没有运行完毕,都会返回true. [注]如果只是要等其他线程运行结束之后再继续操作,可以执行t.join ...

  5. R语言︱情感分析—基于监督算法R语言实现(二)

    每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 笔者寄语:本文大多内容来自未出版的<数据 ...

  6. mysql学习笔记03 mysql数据类型

    数值型:整数型 小数型字符串型时间和日期类型 数值型①整数型1 2 3 4 81bin表示1位,1Byte表示一个字节1B=8b.1汉字=2字节(1 word = 2 byte)1字节=8位(1 by ...

  7. Error Code: 1175. You are using safe update mode and you tried to update a table without a WHERE

    1 错误描述 19:15:34 call sp_store_insert(90) Error Code: 1175. You are using safe update mode and you tr ...

  8. Linux显示USB设备

    Linux显示USB设备 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ lsusb -tv /: Bus 08.Port 1: Dev 1, Class=ro ...

  9. Vue项目搭建及原理三

    我每次写博客都要先在本地写一遍草稿,所以之前有些发布顺序可能会有一丢丢凌乱 哈哈哈,以后绝对改正,那下面我们就说一下创建及项目目录结构吧 三.创建项目 1.初始化Webpack p.p1 { marg ...

  10. 芝麻HTTP: Python爬虫利器之Requests库的用法

    前言 之前我们用了 urllib 库,这个作为入门的工具还是不错的,对了解一些爬虫的基本理念,掌握爬虫爬取的流程有所帮助.入门之后,我们就需要学习一些更加高级的内容和工具来方便我们的爬取.那么这一节来 ...