欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

使用Golang實現(xiàn)加權負載均衡算法的實現(xiàn)代碼

 更新時間:2021年09月09日 17:24:16   作者:Mr.YF  
這篇文章主要介紹了使用Golang實現(xiàn)加權負載均衡算法的實現(xiàn)代碼,詳細說明權重轉發(fā)算法的實現(xiàn),通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

背景描述

如下圖所示,負載均衡做為反向代理,將請求方的請求轉發(fā)至后端的服務節(jié)點,實現(xiàn)服務的請求。

在nginx中可以通過upstream配置server時,設置weight表示對應server的權重。

若存在多個服務節(jié)點時,負載均衡如何通過服務節(jié)點的權重進行轉發(fā)。

如下詳細說明權重轉發(fā)算法的實現(xiàn)。

用三個后端服務節(jié)點為例說明

設置三個后端服務ServerA,ServerB和ServerC,它們的權重分布是 5,3,1

按照加權負載均衡算法,在一輪(5+3+1=9次)中ServerA占5次,ServerB占3次,ServerC占1次,從而實現(xiàn)均衡。

如下圖所示:

為了實現(xiàn)這個功能,可以給每一個后端設置對應的權重5,3,1

變量1:后端服務的權重 Weight

變量2:均衡器累計的總的有效權重EffectiveWeight

變量3:實時統(tǒng)計后端服務的當前權重 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出現(xiàn)了5次,B出現(xiàn)了3次,C出現(xiàn)了1次,滿足三個服務的權重Weight設置。

完成9次獲取后,ABC三個服務的權重都歸0,因此下一輪的9次獲取也是均衡的,

算法實現(xiàn)

按照如上算法說明,使用Golang實現(xiàn)這個算法如下

package weightroundrobin

import (
    "fmt"
    "strings"
)

// 每一個后端服務定義
type BackendServer struct {
    // 實例權重
    Weight int
    // 當前的權重,初始為Weight
    currentWeight int
    // 后端服務名稱
    ServerName string
}

// 通過權重實現(xiàn)調用輪詢的定義
type WeightServerRoundRobin struct {
    // 所有有效的權重總和
    effectiveWeight int
    // 后端服務列表
    backendServerList []*BackendServer
}

// 創(chuàng)建一個負載輪詢器
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實現(xiàn)加權負載均衡算法的文章就介紹到這了,更多相關Golang負載均衡算法內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Go語言中同一個package中函數(shù)互相調用為undefined的解決

    Go語言中同一個package中函數(shù)互相調用為undefined的解決

    這篇文章主要介紹了Go語言中同一個package中函數(shù)互相調用為undefined的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • go?logger不侵入業(yè)務代碼使用slog替換zap并實現(xiàn)callerSkip詳解

    go?logger不侵入業(yè)務代碼使用slog替換zap并實現(xiàn)callerSkip詳解

    這篇文章主要為大家介紹了go?logger不侵入業(yè)務代碼使用slog替換zap并實現(xiàn)callerSkip詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-09-09
  • Go語言多人聊天室項目實戰(zhàn)

    Go語言多人聊天室項目實戰(zhàn)

    這篇文章主要為大家詳細介紹了Go語言多人聊天室項目實戰(zhàn),實現(xiàn)單撩或多撩等多種功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • 詳解Golang中零拷貝的原理以及實踐

    詳解Golang中零拷貝的原理以及實踐

    零拷貝技術相信大家都有所耳聞,但是本文不僅會講述零拷貝技術的原理,并將從實際代碼出發(fā),看看零拷貝技術在golang中的應用,現(xiàn)在讓我們開始吧
    2023-07-07
  • Ubuntu下安裝Go語言開發(fā)環(huán)境及編輯器的相關配置

    Ubuntu下安裝Go語言開發(fā)環(huán)境及編輯器的相關配置

    這篇文章主要介紹了Ubuntu下安裝Go語言開發(fā)環(huán)境及編輯器的相關配置,編輯器方面介紹了包括Vim和Eclipse,需要的朋友可以參考下
    2016-02-02
  • golang獲取變量或對象類型的幾種方式總結

    golang獲取變量或對象類型的幾種方式總結

    在golang中并沒有提供內置函數(shù)來獲取變量的類型,但是通過一定的方式也可以獲取,下面這篇文章主要給大家介紹了關于golang獲取變量或對象類型的幾種方式,需要的朋友可以參考下
    2022-12-12
  • Golang源碼分析之golang/sync之singleflight

    Golang源碼分析之golang/sync之singleflight

    golang/sync庫拓展了官方自帶的sync庫,提供了errgroup、semaphore、singleflight及syncmap四個包,本次先分析第一個包errgroup的源代碼,下面這篇文章主要給大家介紹了關于Golang源碼分析之golang/sync之singleflight的相關資料,需要的朋友可以參考下
    2022-11-11
  • Go批量操作excel導入到mongodb的技巧

    Go批量操作excel導入到mongodb的技巧

    這篇文章主要介紹了Go批量操作excel導入到mongo,包括選擇命令行包,讀取配置連接數(shù)據(jù)庫的方法,本文示例代碼相結合給大家介紹的非常詳細,需要的朋友可以參考下
    2022-03-03
  • golang使用sort接口實現(xiàn)排序示例

    golang使用sort接口實現(xiàn)排序示例

    這篇文章主要介紹了golang使用sort接口實現(xiàn)排序的方法,簡單分析了sort接口的功能并實例演示了基于sort接口的排序實現(xiàn)方法,需要的朋友可以參考下
    2016-07-07
  • Go語言中一定要知道的切片使用注意事項總結

    Go語言中一定要知道的切片使用注意事項總結

    了解和掌握切片的使用注意事項,可以避免意外的程序行為,所以本文就來和大家深入探討一下Go語言切片常見的注意事項,希望對大家有所幫助
    2023-06-06

最新評論