- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章使用Golang实现加权负载均衡算法的实现代码由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
如下图所示,负载均衡做为反向代理,将请求方的请求转发至后端的服务节点,实现服务的请求.
在nginx中可以通过upstream配置server时,设置weight表示对应server的权重.
若存在多个服务节点时,负载均衡如何通过服务节点的权重进行转发.
如下详细说明权重转发算法的实现.
设置三个后端服务servera,serverb和serverc,它们的权重分布是 5,3,1 。
按照加权负载均衡算法,在一轮(5+3+1=9次)中servera占5次,serverb占3次,serverc占1次,从而实现均衡.
如下图所示:
为了实现这个功能,可以给每一个后端设置对应的权重5,3,1 。
变量1:后端服务的权重 weight 。
变量2:均衡器累计的总的有效权重effectiveweight 。
变量3:实时统计后端服务的当前权重 currentweight 。
算法设计 。
第一步,向均衡器中增加后端服务标识 。
第二步,每次获取一个后端服务标识 。
如下是一个一轮(5+3+1=9次)获取的权重变化表:
从这个表中可以看到后端服务轮询的顺序是 a b a c a b a b a,其中a出现了5次,b出现了3次,c出现了1次,满足三个服务的权重weight设置.
完成9次获取后,abc三个服务的权重都归0,因此下一轮的9次获取也是均衡的, 。
按照如上算法说明,使用golang实现这个算法如下 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
|
package weightroundrobin
import
(
"fmt"
"strings"
)
// 每一个后端服务定义
type backendserver struct {
// 实例权重
weight int
// 当前的权重,初始为weight
currentweight int
// 后端服务名称
servername string
}
// 通过权重实现调用轮询的定义
type weightserverroundrobin struct {
// 所有有效的权重总和
effectiveweight int
// 后端服务列表
backendserverlist []*backendserver
}
// 创建一个负载轮询器
func newweightserverroundrobin() *weightserverroundrobin {
return &weightserverroundrobin{
effectiveweight: 0,
}
}
// 增加后端服务名称和权重
func (r *weightserverroundrobin) addbackendserver(backendserver *backendserver) {
r.effectiveweight += backendserver.weight
r.backendserverlist = append(r.backendserverlist, backendserver)
}
// 更具权重获取一个后端服务名称
func (r *weightserverroundrobin) getbackendserver() *backendserver {
var expectbackendserver *backendserver
for _, backendserver := range r.backendserverlist {
// 给每个后端服务增加自身权重
backendserver.currentweight += backendserver.weight
if
expectbackendserver == nil {
expectbackendserver = backendserver
}
if
backendserver.currentweight > expectbackendserver.currentweight {
expectbackendserver = backendserver
}
}
r.visitbackendservercurrentweight()
// 把选择的后端服务权重减掉总权重
expectbackendserver.currentweight -= r.effectiveweight
return expectbackendserver
}
// 打印后端服务的当前权重变化
func (r *weightserverroundrobin) visitbackendservercurrentweight() {
var serverlistforlog []string
for _, backendserver := range r.backendserverlist {
serverlistforlog = append(serverlistforlog,
fmt.sprintf(
"%v"
, backendserver.currentweight))
}
fmt.printf(
"(%v)\n"
, strings.join(serverlistforlog,
", "
))
}
|
写一个单测进行验证 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
package weightroundrobin
import
(
"fmt"
"testing"
)
func testnewweightserverroundrobin(t *testing.t) {
weightserverroundrobin := newweightserverroundrobin()
weightserverroundrobin.addbackendserver(&backendserver{
servername:
"servera"
,
weight: 5,
})
weightserverroundrobin.addbackendserver(&backendserver{
servername:
"serverb"
,
weight: 3,
})
weightserverroundrobin.addbackendserver(&backendserver{
servername:
"serverc"
,
weight: 1,
})
expectservernamelist := []string{
"servera"
,
"serverb"
,
"servera"
,
"serverc"
,
"servera"
,
"serverb"
,
"servera"
,
"serverb"
,
"servera"
,
//
"servera"
,
"serverb"
,
"servera"
,
"serverc"
,
"servera"
,
"serverb"
,
"servera"
,
"serverb"
,
"servera"
,
}
fmt.printf(
"(a, b, c)\n"
)
for ii, expectservername := range expectservernamelist {
weightserverroundrobin.visitbackendservercurrentweight()
backendserver := weightserverroundrobin.getbackendserver()
if
backendserver.servername != expectservername {
t.errorf(
"%v.%v.expect:%v, actual:%v"
, t.name(), ii, expectservername, backendserver.servername)
return
}
}
}
|
运行单元测试,观察运行结果是否符合算法设计的预期 。
=== run testnewweightserverroundrobin (a, b, c) (0, 0, 0) (5, 3, 1) (-4, 3, 1) (1, 6, 2) (1, -3, 2) (6, 0, 3) (-3, 0, 3) (2, 3, 4) (2, 3, -5) (7, 6, -4) (-2, 6, -4) (3, 9, -3) (3, 0, -3) (8, 3, -2) (-1, 3, -2) (4, 6, -1) (4, -3, -1) (9, 0, 0) --- pass: testnewweightserverroundrobin (0.00s) pass 。
参考材料:
https://github.com/phusion/nginx/commit/27e94984486058d73157038f7950a0a36ecc6e35 。
到此这篇关于使用golang实现加权负载均衡算法的文章就介绍到这了,更多相关golang负载均衡算法内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://www.cnblogs.com/voipman/p/15020713.html 。
最后此篇关于使用Golang实现加权负载均衡算法的实现代码的文章就讲到这里了,如果你想了解更多关于使用Golang实现加权负载均衡算法的实现代码的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我在网上搜索但没有找到任何合适的文章解释如何使用 javascript 使用 WCF 服务,尤其是 WebScriptEndpoint。 任何人都可以对此给出任何指导吗? 谢谢 最佳答案 这是一篇关于
我正在编写一个将运行 Linux 命令的 C 程序,例如: cat/etc/passwd | grep 列表 |剪切-c 1-5 我没有任何结果 *这里 parent 等待第一个 child (chi
所以我正在尝试处理文件上传,然后将该文件作为二进制文件存储到数据库中。在我存储它之后,我尝试在给定的 URL 上提供文件。我似乎找不到适合这里的方法。我需要使用数据库,因为我使用 Google 应用引
我正在尝试制作一个宏,将下面的公式添加到单元格中,然后将其拖到整个列中并在 H 列中复制相同的公式 我想在 F 和 H 列中输入公式的数据 Range("F1").formula = "=IF(ISE
问题类似于this one ,但我想使用 OperatorPrecedenceParser 解析带有函数应用程序的表达式在 FParsec . 这是我的 AST: type Expression =
我想通过使用 sequelize 和 node.js 将这个查询更改为代码取决于在哪里 select COUNT(gender) as genderCount from customers where
我正在使用GNU bash,版本5.0.3(1)-发行版(x86_64-pc-linux-gnu),我想知道为什么简单的赋值语句会出现语法错误: #/bin/bash var1=/tmp
这里,为什么我的代码在 IE 中不起作用。我的代码适用于所有浏览器。没有问题。但是当我在 IE 上运行我的项目时,它发现错误。 而且我的 jquery 类和 insertadjacentHTMl 也不
我正在尝试更改标签的innerHTML。我无权访问该表单,因此无法编辑 HTML。标签具有的唯一标识符是“for”属性。 这是输入和标签的结构:
我有一个页面,我可以在其中返回用户帖子,可以使用一些 jquery 代码对这些帖子进行即时评论,在发布新评论后,我在帖子下插入新评论以及删除 按钮。问题是 Delete 按钮在新插入的元素上不起作用,
我有一个大约有 20 列的“管道分隔”文件。我只想使用 sha1sum 散列第一列,它是一个数字,如帐号,并按原样返回其余列。 使用 awk 或 sed 执行此操作的最佳方法是什么? Accounti
我需要将以下内容插入到我的表中...我的用户表有五列 id、用户名、密码、名称、条目。 (我还没有提交任何东西到条目中,我稍后会使用 php 来做)但由于某种原因我不断收到这个错误:#1054 - U
所以我试图有一个输入字段,我可以在其中输入任何字符,但然后将输入的值小写,删除任何非字母数字字符,留下“。”而不是空格。 例如,如果我输入: 地球的 70% 是水,-!*#$^^ & 30% 土地 输
我正在尝试做一些我认为非常简单的事情,但出于某种原因我没有得到想要的结果?我是 javascript 的新手,但对 java 有经验,所以我相信我没有使用某种正确的规则。 这是一个获取输入值、检查选择
我想使用 angularjs 从 mysql 数据库加载数据。 这就是应用程序的工作原理;用户登录,他们的用户名存储在 cookie 中。该用户名显示在主页上 我想获取这个值并通过 angularjs
我正在使用 autoLayout,我想在 UITableViewCell 上放置一个 UIlabel,它应该始终位于单元格的右侧和右侧的中心。 这就是我想要实现的目标 所以在这里你可以看到我正在谈论的
我需要与 MySql 等效的 elasticsearch 查询。我的 sql 查询: SELECT DISTINCT t.product_id AS id FROM tbl_sup_price t
我正在实现代码以使用 JSON。 func setup() { if let flickrURL = NSURL(string: "https://api.flickr.com/
我尝试使用for循环声明变量,然后测试cols和rols是否相同。如果是,它将运行递归函数。但是,我在 javascript 中执行 do 时遇到问题。有人可以帮忙吗? 现在,在比较 col.1 和
我举了一个我正在处理的问题的简短示例。 HTML代码: 1 2 3 CSS 代码: .BB a:hover{ color: #000; } .BB > li:after {
我是一名优秀的程序员,十分优秀!