使用Golang实现加权负载均衡算法的实现代码
本篇文章给大家分享《使用Golang实现加权负载均衡算法的实现代码》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。
背景描述
如下图所示,负载均衡做为反向代理,将请求方的请求转发至后端的服务节点,实现服务的请求。
在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
算法设计
第一步,向均衡器中增加后端服务标识
- 将三个后端服务标识和权重Weight增加到负载均衡器列表中。
- 每次增加后端服务时,累计总的有效权重EffectiveWeight。
第二步,每次获取一个后端服务标识
- 对均衡器中的所有后端服务增加自己的权重Weight,即(5,3,1),计算ABC三个服务的当前权重。
- 选择当前权重CurrentWeight最大的服务,做为本次期望的后端服务。
- 将期望的后端服务的当前权重CurrentWeight减小总的权重EffectiveWeight,供下一轮使用。
如下是一个一轮(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实现这个算法如下
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, ", ")) }
写一个单测进行验证
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学习网公众号,一起学习编程~

- 上一篇
- 手把手带你走进Go语言之语法基础解析

- 下一篇
- Go应用中优雅处理Error的技巧总结
-
- 细心的冬瓜
- 很详细,码住,感谢大佬的这篇文章内容,我会继续支持!
- 2023-04-03 04:51:57
-
- 火星上的含羞草
- 很详细,收藏了,感谢师傅的这篇文章内容,我会继续支持!
- 2023-02-18 17:07:19
-
- 结实的乌冬面
- 写的不错,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢师傅分享文章!
- 2023-02-12 09:42:45
-
- 感动的发夹
- 写的不错,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢楼主分享技术贴!
- 2023-02-09 04:34:46
-
- 欢呼的台灯
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者大大分享博文!
- 2023-02-07 17:13:46
-
- 懦弱的睫毛膏
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢老哥分享技术文章!
- 2023-02-02 17:21:32
-
- 纯情的热狗
- 这篇文章内容太及时了,很详细,太给力了,已收藏,关注楼主了!希望楼主能多写Golang相关的文章。
- 2023-01-19 00:52:10
-
- 微笑的长颈鹿
- 这篇文章太及时了,很详细,太给力了,收藏了,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-01-06 22:03:23
-
- 柔弱的钥匙
- 这篇博文出现的刚刚好,太详细了,真优秀,收藏了,关注作者大大了!希望作者大大能多写Golang相关的文章。
- 2023-01-06 17:56:42
-
- 可靠的钢铁侠
- 这篇技术贴太及时了,up主加油!
- 2023-01-06 17:23:16
-
- 欣慰的鼠标
- 这篇技术文章太及时了,细节满满,真优秀,已加入收藏夹了,关注师傅了!希望师傅能多写Golang相关的文章。
- 2023-01-02 09:10:25
-
- Golang · Go教程 | 4小时前 |
- DebianOpenSSL安装失败的终极解决方案
- 501浏览 收藏
-
- Golang · Go教程 | 5小时前 |
- Debian数据快速提取技巧
- 216浏览 收藏
-
- Golang · Go教程 | 8小时前 |
- Debian系统JS依赖管理终极攻略
- 218浏览 收藏
-
- Golang · Go教程 | 9小时前 |
- Debian上Hadoop作业调度实用技巧
- 100浏览 收藏
-
- Golang · Go教程 | 10小时前 |
- Go语言闭包误区与匿名函数深度解析
- 222浏览 收藏
-
- Golang · Go教程 | 10小时前 |
- Debian系统安全回收数据的正确攻略
- 111浏览 收藏
-
- Golang · Go教程 | 12小时前 |
- Debian高效fetch技巧与使用攻略
- 125浏览 收藏
-
- Golang · Go教程 | 18小时前 |
- Debian邮件服务器升级维护攻略
- 474浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 笔灵AI生成答辩PPT
- 探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
- 14次使用
-
- 知网AIGC检测服务系统
- 知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
- 22次使用
-
- AIGC检测-Aibiye
- AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
- 30次使用
-
- 易笔AI论文
- 易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
- 40次使用
-
- 笔启AI论文写作平台
- 笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
- 35次使用
-
- goHTTP2的头部压缩算法hpack实现详解
- 2022-12-22 398浏览
-
- go语言算法题解二叉树的最小深度
- 2022-12-22 327浏览
-
- Go 语言简单实现Vigenere加密算法
- 2022-12-29 319浏览
-
- 请教go语言算法将二维数组转换为目录结构
- 2023-01-07 272浏览
-
- 为上岸Alibaba,我把Github上Java面试题都整理了一遍
- 2023-02-24 130浏览