负载均衡算法模块主要的功能是从负载均衡器中获取服务器列表信息,根据算法选取出一个服务器。

IRule

  负载均衡算法接口

  1. public interface IRule{
  2. public Server choose(Object key);//选择一个服务器
  3. public void setLoadBalancer(ILoadBalancer lb);//设置负载均衡器
  4. public ILoadBalancer getLoadBalancer(); //获取负载均衡器
  5. }

  通过BaseLoadBalancer的setRule或构造函数来为BaseLoadBalancer添加IRule

  1. public void setRule(IRule rule) {
  2. if (rule != null) {
  3. this.rule = rule;
  4. } else {
  5. /* default rule */
  6. this.rule = new RoundRobinRule();
  7. }
  8. if (this.rule.getLoadBalancer() != this) {
  9. this.rule.setLoadBalancer(this);
  10. }
  11. }

RandomRule

  生成一个随机数,从负载均衡器中选取一个服务器。

  1. public Server choose(ILoadBalancer lb, Object key) {
  2. if (lb == null) {
  3. return null;
  4. }
  5. Server server = null;
  6. while (server == null) {
  7. if (Thread.interrupted()) {
  8. return null;
  9. }
  10. List<Server> upList = lb.getReachableServers();
  11. List<Server> allList = lb.getAllServers();
  12. int serverCount = allList.size();
  13. if (serverCount == 0) {
  14. return null;
  15. }
  16. int index = rand.nextInt(serverCount);
  17. server = upList.get(index);
  18. if (server == null) {
  19. Thread.yield();
  20. continue;
  21. }
  22. if (server.isAlive()) {
  23. return (server);
  24. }
  25. server = null;
  26. Thread.yield();
  27. }
  28. return server;
  29. }

RoundRobinRule

  轮询从负载均衡器中选取一个服务器。

  1. public Server choose(ILoadBalancer lb, Object key) {
  2. if (lb == null) {
  3. log.warn("no load balancer");
  4. return null;
  5. }
  6. Server server = null;
  7. int count = 0;
  8. while (server == null && count++ < 10) {
  9. List<Server> reachableServers = lb.getReachableServers();
  10. List<Server> allServers = lb.getAllServers();
  11. int upCount = reachableServers.size();
  12. int serverCount = allServers.size();
  13.  
  14. if ((upCount == 0) || (serverCount == 0)) {
  15. log.warn("No up servers available from load balancer: " + lb);
  16. return null;
  17. }
  18. int nextServerIndex = incrementAndGetModulo(serverCount);
  19. server = allServers.get(nextServerIndex);
  20. if (server == null) {
  21. /* Transient. */
  22. Thread.yield();
  23. continue;
  24. }
  25. if (server.isAlive() && (server.isReadyToServe())) {
  26. return (server);
  27. }
  28. server = null;
  29. }
  30. if (count >= 10) {
  31. log.warn("No available alive servers after 10 tries from load balancer: "
  32. + lb);
  33. }
  34. return server;
  35. }

BestAvailableRule

  选择并发量最小且没有被熔断的服务器,需要使用到LoadBalancerStats来获取服务器的状态。

  1. public Server choose(Object key) {
  2. if (loadBalancerStats == null) {
  3. return super.choose(key);
  4. }
  5. List<Server> serverList = getLoadBalancer().getAllServers();
  6. int minimalConcurrentConnections = Integer.MAX_VALUE;
  7. long currentTime = System.currentTimeMillis();
  8. Server chosen = null;
  9. for (Server server: serverList) {
  10. ServerStats serverStats = loadBalancerStats.getSingleServerStat(server);
  11. if (!serverStats.isCircuitBreakerTripped(currentTime)) {
  12. int concurrentConnections = serverStats.getActiveRequestsCount(currentTime);
  13. if (concurrentConnections < minimalConcurrentConnections) {
  14. minimalConcurrentConnections = concurrentConnections;
  15. chosen = server;
  16. }
  17. }
  18. }
  19. if (chosen == null) {
  20. return super.choose(key);
  21. } else {
  22. return chosen;
  23. }
  24. }

WeightedResponseTimeRule

  按照响应时间的比例来选择服务器。首先内部会有一个定时器,定时从负载均衡器里面读取服务器的平均响应时间,然后根据平均响应时间转换成权重。

  1. class DynamicServerWeightTask extends TimerTask {
  2. public void run() {
  3. ServerWeight serverWeight = new ServerWeight();
  4. try {
  5. serverWeight.maintainWeights();
  6. } catch (Exception e) {
  7. logger.error("Error running DynamicServerWeightTask for {}", name, e);
  8. }
  9. }
  10. }
  11. class ServerWeight {
  12. public void maintainWeights() {
  13. ILoadBalancer lb = getLoadBalancer();
  14. if (lb == null) {
  15. return;
  16. }
  17. if (!serverWeightAssignmentInProgress.compareAndSet(false, true)) {
  18. return;
  19. }
  20. try {
  21. logger.info("Weight adjusting job started");
  22. AbstractLoadBalancer nlb = (AbstractLoadBalancer) lb;
  23. LoadBalancerStats stats = nlb.getLoadBalancerStats();
  24. if (stats == null) {
  25. return;
  26. }
  27. double totalResponseTime = 0;
  28. // find maximal 95% response time
  29. for (Server server : nlb.getAllServers()) {
  30. // this will automatically load the stats if not in cache
  31. ServerStats ss = stats.getSingleServerStat(server);
  32. totalResponseTime += ss.getResponseTimeAvg();
  33. }
  34. // weight for each server is (sum of responseTime of all servers - responseTime)
  35. // so that the longer the response time, the less the weight and the less likely to be chosen
  36. Double weightSoFar = 0.0;
  37. // create new list and hot swap the reference
  38. List<Double> finalWeights = new ArrayList<Double>();
  39. for (Server server : nlb.getAllServers()) {
  40. ServerStats ss = stats.getSingleServerStat(server);
  41. double weight = totalResponseTime - ss.getResponseTimeAvg();
  42. weightSoFar += weight;
  43. finalWeights.add(weightSoFar);
  44. }
  45. setWeights(finalWeights);
  46. } catch (Exception e) {
  47. logger.error("Error calculating server weights", e);
  48. } finally {
  49. serverWeightAssignmentInProgress.set(false);
  50. }
  51.  
  52. }
  53. }

  然后根据权重来选择服务器

  1. public Server choose(ILoadBalancer lb, Object key) {
  2. if (lb == null) {
  3. return null;
  4. }
  5. Server server = null;
  6. while (server == null) {
  7. // get hold of the current reference in case it is changed from the other thread
  8. List<Double> currentWeights = accumulatedWeights;
  9. if (Thread.interrupted()) {
  10. return null;
  11. }
  12. List<Server> allList = lb.getAllServers();
  13. int serverCount = allList.size();
  14. if (serverCount == 0) {
  15. return null;
  16. }
  17. int serverIndex = 0;
  18. // last one in the list is the sum of all weights
  19. double maxTotalWeight = currentWeights.size() == 0 ? 0 : currentWeights.get(currentWeights.size() - 1);
  20. // No server has been hit yet and total weight is not initialized
  21. // fallback to use round robin
  22. if (maxTotalWeight < 0.001d) {
  23. server = super.choose(getLoadBalancer(), key);
  24. if(server == null) {
  25. return server;
  26. }
  27. } else {
  28. // generate a random weight between 0 (inclusive) to maxTotalWeight (exclusive)
  29. double randomWeight = random.nextDouble() * maxTotalWeight;
  30. // pick the server index based on the randomIndex
  31. int n = 0;
  32. for (Double d : currentWeights) {
  33. if (d >= randomWeight) {
  34. serverIndex = n;
  35. break;
  36. } else {
  37. n++;
  38. }
  39. }
  40. server = allList.get(serverIndex);
  41. }
  42.  
  43. if (server == null) {
  44. /* Transient. */
  45. Thread.yield();
  46. continue;
  47. }
  48.  
  49. if (server.isAlive()) {
  50. return (server);
  51. }
  52.  
  53. // Next.
  54. server = null;
  55. }
  56. return server;
  57. }

AvailabilityFilteringRule

  使用RoundRobinRule来选择服务器,并且通过AvailabilityPredicate进行筛选。AvailabilityPredicate会剔除熔断的和超过指定并发量的server。

  1. public Server choose(Object key) {
  2. int count = 0;
  3. Server server = roundRobinRule.choose(key);
  4. while (count++ <= 10) {
  5. if (predicate.apply(new PredicateKey(server))) {
  6. return server;
  7. }
  8. server = roundRobinRule.choose(key);
  9. }
  10. return super.choose(key);
  11. }

  AvailabilityPredicate:

  1. public boolean apply(@Nullable PredicateKey input) {
  2. LoadBalancerStats stats = getLBStats();
  3. if (stats == null) {
  4. return true;
  5. }
  6. return !shouldSkipServer(stats.getSingleServerStat(input.getServer()));
  7. }
  8. private boolean shouldSkipServer(ServerStats stats) {
  9. if ((CIRCUIT_BREAKER_FILTERING.get() && stats.isCircuitBreakerTripped())
  10. || stats.getActiveRequestsCount() >= activeConnectionsLimit.get()) {
  11. return true;
  12. }
  13. return false;
  14. }

  使用AvailabilityFilteringRule涉及配置:

属性 实现 默认值
niws.loadbalancer.availabilityFilteringRule.filterCircuitTripped  是否剔除熔断server true

niws.loadbalancer.availabilityFilteringRule.activeConnectionsLimit

最大连接数 Integer.MAX_VALUE

ZoneAvoidanceRule

  这个rule虽然继承了PredicateBasedRule但是在使用上都与上面的rule不一样,其实他的核心主要是为ZoneAwareLoadBalancer提供了筛选zone的静态方法,他并不通用。

  静态方法getAvailableZones,会遍历所有的zone,以zone为单位,检查各个zone的实例个数,熔断比率,来决定是否包含改zone。

  静态方法createSnapshot,将LoadBalancerStats按zone返回map结构

类图

Predicate

  用于过滤服务器,ribbon提供了三个过滤条件,AvailabilityPredicate、ZoneAvoidancePredicate、ZoneAffinityPredicate。PredicateKey为过滤的参数。

ribbon源码(4) 载均衡算法的更多相关文章

  1. Spring Cloud Ribbon 源码分析---负载均衡算法

    上一篇分析了Ribbon如何发送出去一个自带负载均衡效果的HTTP请求,本节就重点分析各个算法都是如何实现. 负载均衡整体是从IRule进去的: public interface IRule{ /* ...

  2. Spring Cloud Ribbon源码分析---负载均衡实现

    上一篇结合 Eureka 和 Ribbon 搭建了服务注册中心,利用Ribbon实现了可配置负载均衡的服务调用.这一篇我们来分析Ribbon实现负载均衡的过程. 从 @LoadBalanced入手 还 ...

  3. ribbon源码(2) 负载均衡器

    负载均衡器对外提供负载均衡的功能,本质上是是维护当前服务的服务器列表和服务器状态,通过负载均衡算法选取合适的服务器地址. 用户可以通过实现ILoadBalancer来实现自己的负载均衡器,ribbon ...

  4. Ribbon源码分析(二)-- 服务列表的获取和负载均衡算法分析

    上一篇博客(https://www.cnblogs.com/yangxiaohui227/p/12614343.html)分享了ribbon如何实现对http://product/info/这个链接重 ...

  5. 【一起学源码-微服务】Ribbon 源码四:进一步探究Ribbon的IRule和IPing

    前言 前情回顾 上一讲深入的讲解了Ribbon的初始化过程及Ribbon与Eureka的整合代码,与Eureka整合的类就是DiscoveryEnableNIWSServerList,同时在Dynam ...

  6. 【一起学源码-微服务】Ribbon源码五:Ribbon源码解读汇总篇~

    前言 想说的话 [一起学源码-微服务-Ribbon]专栏到这里就已经全部结束了,共更新四篇文章. Ribbon比较小巧,这里是直接 读的spring cloud 内嵌封装的版本,里面的各种config ...

  7. ribbon源码(1) 概述

    ribbon的核心功能是提供客户端在进行网络请求时负载均衡的能力.主要有以下几个模块: 负载均衡器模块 负载均衡器模块提供了负载均衡能力,详细参见ribbon源码之负载均衡器. 配置模块 配置模块管理 ...

  8. Android开源项目 Universal imageloader 源码研究之Lru算法

    https://github.com/nostra13/Android-Universal-Image-Loader universal imageloader 源码研究之Lru算法 LRU - Le ...

  9. Spring源码加载过程图解(一)

    最近看了一下Spring源码加载的简装版本,为了更好的理解,所以在绘图的基础上,进行了一些总结.(图画是为了理解和便于记忆Spring架构) Spring的核心是IOC(控制反转)和AOP(面向切面编 ...

随机推荐

  1. Kubernetes 的层级命名空间介绍

    原文链接:https://fuckcloudnative.io/posts/introducing-hierarchical-namespaces/ 在单个 Kubernetes 集群上安全托管大量用 ...

  2. 实现0.5px边框线

    实现0.5px边框方法 方案一:利用渐变(原理:高度1px,背景渐变,一半有颜色,一半透明) CSS部分 .container { width: 500px; margin: 0px auto; } ...

  3. Collections.synchronizedMap()与ConcurrentHashMap区别

    Collections.synchronizedMap()与ConcurrentHashMap主要区别是:Collections.synchronizedMap()和Hashtable一样,实现上在调 ...

  4. Wireshark中的Checksum: 0x90c5 [validation disabled]问题

    Wireshark中的Checksum: 0x90c5 [validation disabled]问题 废话不多说先上问题图: 这是我在做关于DNS协议PPT的时候出现的协议树第五项展开结果,可以发现 ...

  5. VyOS软路由系统基本设置

    1. VyOS简介 VyOS是一个开源的网络操作系统,可以安装在物理硬件上,也可以安装在你自己的虚拟机上,或者是一个云平台上.它基于GNU/Linux,并加入了多个应用程序,如:Quagga, ISC ...

  6. 团队作业4:第三篇Scrum冲刺博客(歪瑞古德小队)

    目录 一.Daily Scrum Meeting 1.1 会议照片 1.2 项目进展 二.项目燃尽图 三.签入记录 3.1 代码/文档签入记录 3.2 Code Review 记录 3.3 issue ...

  7. Java多线程_同步工具CyclicBarrier

    CyclicBarrier概念:CyclicBarrier是多线程中的一个同步工具,它允许一组线程互相等待,直到到达某个公共屏障点.形象点儿说,CyclicBarrier就是一个屏障,要求这一组线程中 ...

  8. UnitTest单元测试框架解析【实用篇】

    UnitTest是展开自动化测试的基础——这个框架很重要!首先我们先自己写一个测试类: 1.被测试类 Widthget.py: # coding: utf-8class Widthget: def _ ...

  9. python os库的使用方法 + 自动化安装第三方库脚本

    一.os库基本介绍 os库提供通用的.基本的操作系统交互功能,包括windows.Mac os.linux os库是python标准库,包含几百个函数 常用路径操作.进程管理.环境参数等几类 路径操作 ...

  10. CefSharp如何判断页面是否加载完

    问题:CefSharp如何判断页面是否加载完毕. 摘要:相信C#用CefSharp做浏览器来发的应该有很多人都会有遇到这个问题.特别是要执行JavaScript的时候,涉及到跨页面的JavaScrip ...