2019年1月8日16:10:51

svn地址:svn://gitee.com/zxadmin/live_z    代码在code里面

<?php

/*
* 加权轮训算法
*
*
* $arr = array(
array('id' => 'A', 'weight' => 3),
array('id' => 'B', 'weight' => 3),
array('id' => 'C', 'weight' => 6),
array('id' => 'D', 'weight' => 4),
array('id' => 'E', 'weight' => 1),
);
* $arr = array(
array('id' => '192.168.1.1', 'weight' => 3),
array('id' => '192.168.1.2', 'weight' => 3),
array('id' => '192.168.1.3', 'weight' => 6),
array('id' => '192.168.1.4', 'weight' => 4),
array('id' => '192.168.1.5', 'weight' => 1),
);
*/ class ZxWeightedRoundRobin { private static $weightArray = array();
private static $currentIndex = -1; //代表上一次选择的服务器
private static $gcd; //表示集合S中所有服务器权值的最大公约数
private static $currentWeight = 0; //当前调度的权值
private static $maxWeight; //最大元素的值
private static $count; //总元素个数 public function __construct(array $weightArray) {
self::$weightArray = $weightArray;
self::$gcd = self::getGcd(self::$weightArray);
// p(self::$gcd);
} /*
*
* 算法的原理是:在服务器数组S中,首先计算所有服务器权重的最大值max(S),以及所有服务器权重的最大公约数gcd(S)。
index表示本次请求到来时,选择的服务器的索引,初始值为-1;current_weight表示当前调度的权值,初始值为max(S)。
当请求到来时,从index+1开始轮询服务器数组S,找到其中权重大于current_weight的第一个服务器,用于处理该请求。记录其索引到结果序列中。
在轮询服务器数组时,如果到达了数组末尾,则重新从头开始搜索,并且减小current_weight的值:current_weight -= gcd(S)。如果current_weight等于0,则将其重置为max(S)。
* 参考博客:https://blog.csdn.net/gqtcgq/article/details/52076997
* https://blog.csdn.net/jjavaboy/article/details/45604569
*/ public function getWeight() {
while (true) {
self::$currentIndex = ((int) self::$currentIndex + 1) % (int) self::$count;
if (self::$currentIndex == 0) {
self::$currentWeight = (int) self::$currentWeight - (int) self::$gcd;
if (self::$currentWeight <= 0) {
self::$currentWeight = (int) self::$maxWeight;
if (self::$currentWeight == 0) {
return null;
}
}
}
// p(self::$currentIndex);
if ((int) (self::$weightArray[self::$currentIndex]['weight']) >= self::$currentWeight) {
return self::$weightArray[self::$currentIndex];
}
}
} //获取最大公约数 Greatest common divisor 最大共同被除数
private static function getGcd(array $weightArray) {
if (empty($weightArray) || !is_array($weightArray)) {
throw new \Exception('数组不能为空');
}
$weight = [];
//权重只能为正整数
foreach ($weightArray as $k => $v) {
if (!is_int($v['weight']) || $v['weight'] <= 0) {
throw new \Exception('权限不合法');
}
$weight[] = $v['weight'];
}
$min = min($weight);
self::$maxWeight = max($weight);
self::$count = count($weight);
//如果最小值是1,最小公约数就必定是1
if ($min == 1) {
return 1;
}
//如果不是1,就每个元素,循环查询对最小值往下做整除处理,如果全可以整除,如果有一个不能就中断
for ($i = $min; $i > 1; $i--) {
foreach ($weight as $k1 => $v1) {
if ($v1 % $i == 0) {
$status = true;
} else {
$status = false;
break;
}
}
if ($status) {
return $i;
} else {
return 1;
}
}
} }

这个方法,我理解了最大公约数,但是 getWeight 方法还没有彻底理解,最小公约的话,就只需要修改

for ($i = $min; $i > 1; $i--) {遍历就可以

测试调用方法
$arr = array(
array('id' => 'A', 'weight' => 3),
array('id' => 'B', 'weight' => 3),
array('id' => 'C', 'weight' => 6),
array('id' => 'D', 'weight' => 4),
array('id' => 'E', 'weight' => 2),
);
$weight = new ZxWeightedRoundRobin($arr);
$a = 0;
$b = 0;
$c = 0;
$d = 0;
$e = 0;
for ($j = 0; $j < 100; $j++) {
$weightInfo = $weight->getWeight();
print_r($weightInfo);
echo $weightInfo['id'] . '----------------------weight:' . $weightInfo['weight'] . '<br/>';
if ($weightInfo['id'] == 'A') {
$a++;
}
if ($weightInfo['id'] == 'B') {
$b++;
}
if ($weightInfo['id'] == 'C') {
$c++;
}
if ($weightInfo['id'] == 'D') {
$d++;
}
if ($weightInfo['id'] == 'E') {
$e++;
}
}
echo 'A:' . $a . '<br/>';
echo 'B:' . $b . '<br/>';
echo 'C:' . $c . '<br/>';
echo 'D:' . $d . '<br/>';
echo 'E:' . $e . '<br/>';
exit;
 

PHP算法学习(2) 轮训加权算法的更多相关文章

  1. 四旋翼基础算法学习2-IMU输入滤波算法

    前言: 处理器读取陀螺仪加速度计数据后首先需要对数据进行滤波处理,此文分析比较几种常用的滤波算法. 参考学习:四轴加速度计滤波 IMU: IMU使用MPU9250(即MPU6500),设置加速度量程± ...

  2. 【算法学习笔记】Meissel-Lehmer 算法 (亚线性时间找出素数个数)

    「Meissel-Lehmer 算法」是一种能在亚线性时间复杂度内求出 \(1\sim n\) 内质数个数的一种算法. 在看素数相关论文时发现了这个算法,论文链接:Here. 算法的细节来自 OI w ...

  3. Kmeans算法学习与SparkMlLib Kmeans算法尝试

    K-means算法是最为经典的基于划分的聚类方法,是十大经典数据挖掘算法之一.K-means算法的基本思想是:以空间中k个点为中心进行聚类,对最靠近他们的对象归类.通过迭代的方法,逐次更新各聚类中心的 ...

  4. 算法学习之选择排序算法的python实现

    ——参考自<算法图解> def findSmallest(arr): # 假设第一个元素最小 smallest = arr[0] smallest_index = 0 for i in r ...

  5. 算法学习之二分查找算法的python实现

    ——参考自<算法图解> 我们假设需要查找的数组是有序的(从大到小或者从小到大),如果无序,可以在第四行后插入一句 my_list.sort() 完整代码如下 def binary_sear ...

  6. 算法学习笔记:Kosaraju算法

    Kosaraju算法一看这个名字很奇怪就可以猜到它也是一个根据人名起的算法,它的发明人是S. Rao Kosaraju,这是一个在图论当中非常著名的算法,可以用来拆分有向图当中的强连通分量. 背景知识 ...

  7. 算法学习笔记:Tarjan算法

    在上一篇文章当中我们分享了强连通分量分解的一个经典算法Kosaraju算法,它的核心原理是通过将图翻转,以及两次递归来实现.今天介绍的算法名叫Tarjan,同样是一个很奇怪的名字,奇怪就对了,这也是以 ...

  8. Python之路,Day21 - 常用算法学习

    Python之路,Day21 - 常用算法学习   本节内容 算法定义 时间复杂度 空间复杂度 常用算法实例 1.算法定义 算法(Algorithm)是指解题方案的准确而完整的描述,是一系列解决问题的 ...

  9. 数据挖掘算法学习(八)Adaboost算法

    本文不定期更新.原创文章,转载请附上链接http://blog.csdn.net/iemyxie/article/details/40423907 谢谢 Adaboost是一种迭代算法,其核心思想是针 ...

随机推荐

  1. 2.10 while循环应用

    while循环应用 1. 计算1~100的累积和(包含1和100) 参考代码如下: #encoding=utf-8 i = 1 sum = 0 while i <= 100: sum = sum ...

  2. Apache Ambari安装过程(CentOS 6.5)

    一.准备环境 1.host 本人准备了三台服务器, vim /etc/hosts 192.168.1.131 dk11 192.168.1.132 dk21 192.168.1.133 dk31 2. ...

  3. Element老司机开车了

    orm Select选择器   坑: 1.选择器视图层一直渲染最后一个元素(value-key作为唯一标识符是关键)2.视图不更新,没有在data函数中声明变量,或者其他地方重置了已经声明过得变量 o ...

  4. Faster RCNN 学习笔记

    下面的介绍都是基于VGG16 的Faster RCNN网络,各网络的差异在于Conv layers层提取特征时有细微差异,至于后续的RPN层.Pooling层及全连接的分类和目标定位基本相同. 一). ...

  5. spring cloud 微服务调用--ribbon和feign调用

    这里介绍ribbon和feign调用两种通信服务调用方式,同时介绍如何引入第三方服务调用.案例包括了ribbon负载均衡和hystrix熔断--服务降级的处理,以及feign声明式服务调用.例子包括s ...

  6. LOJ 3049: 洛谷 P5284: 「十二省联考 2019」字符串问题

    题目传送门:LOJ #3049. 题意简述: 给定一个长度为 \(n\) 的母串 \(S\). 有 \(n_a\) 个 A 类串,都是 \(S\) 的子串,以区间的形式给出. 有 \(n_b\) 个 ...

  7. 51nod--1183 编辑距离(动态规划)

    题目: 1183 编辑距离 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 编辑距离,又称Levenshtein距离(也叫做Edit Distance),是指 ...

  8. iOS -- Effective Objective-C 阅读笔记 (8)

    若想令自己缩写的对象具有拷贝功能, 则需要实现 NSCopying 协议, 如果自定义的对象分为可变版本与不可变版本, 那么就要同时实现 NSCopying 协议和 NSMutableCopying ...

  9. MySQL基础使用

    数据库 其实我们常常说的数据库,应该叫数据库系统. 表和库 数据表:用来保存数据的表格 数据库:用来统一管理数据表的容器 启动mysql 关闭mysql service mysqld start(启动 ...

  10. 《剑指offer》整数中1出现的次数

    本题来自<剑指offer> 反转链表 题目: 思路: C++ Code: Python Code: 总结: