golang grpc 負(fù)載均衡的方法
微服務(wù)架構(gòu)里面,每個服務(wù)都會有很多節(jié)點(diǎn),如果流量分配不均勻,會造成資源的浪費(fèi),甚至將一些機(jī)器壓垮,這個時候就需要負(fù)載均衡,最簡單的一種策略就是輪詢,順序依次選擇不同的節(jié)點(diǎn)訪問。
grpc 在客戶端提供了負(fù)載均衡的實(shí)現(xiàn),并提供了服務(wù)地址解析和更新的接口(默認(rèn)提供了 DNS 域名解析的支持),方便不同服務(wù)的集成
使用示例
conn, err := grpc.Dial(
"",
grpc.WithInsecure(),
// 負(fù)載均衡,使用 consul 作服務(wù)發(fā)現(xiàn)
grpc.WithBalancer(grpc.RoundRobin(grpclb.NewConsulResolver(
"127.0.0.1:8500", "grpc.health.v1.add",
))),
)
創(chuàng)建連接的時候可以使用 WithBalancer 選項(xiàng)來指定負(fù)載均衡策略,這里使用 RoundRobin 算法,其實(shí)就是輪詢策略
與 consul 的集成
有了負(fù)載均衡策略,還需要一個地址解析和更新策略,可以使用 DNS 服務(wù)來實(shí)現(xiàn),但如果我們使用 consul 來做服務(wù)的注冊和發(fā)現(xiàn),可以通過實(shí)現(xiàn) ‘naming.Resolver' 和 ‘naming.Watcher' 接口來支持
- naming.Resolver: 實(shí)現(xiàn)地址解析
- naming.Watcher: 實(shí)現(xiàn)節(jié)點(diǎn)的變更,添加或者刪除
func NewConsulResolver(address string, service string) naming.Resolver {
return &consulResolver{
address: address,
service: service,
}
}
type consulResolver struct {
address string
service string
}
func (r *consulResolver) Resolve(target string) (naming.Watcher, error) {
config := api.DefaultConfig()
config.Address = r.address
client, err := api.NewClient(config)
if err != nil {
return nil, err
}
return &consulWatcher{
client: client,
service: r.service,
addrs: map[string]struct{}{},
}, nil
}
type consulWatcher struct {
client *api.Client
service string
addrs map[string]struct{}
lastIndex uint64
}
func (w *consulWatcher) Next() ([]*naming.Update, error) {
for {
services, metainfo, err := w.client.Health().Service(w.service, "", true, &api.QueryOptions{
WaitIndex: w.lastIndex, // 同步點(diǎn),這個調(diào)用將一直阻塞,直到有新的更新
})
if err != nil {
logrus.Warn("error retrieving instances from Consul: %v", err)
}
w.lastIndex = metainfo.LastIndex
addrs := map[string]struct{}{}
for _, service := range services {
addrs[net.JoinHostPort(service.Service.Address, strconv.Itoa(service.Service.Port))] = struct{}{}
}
var updates []*naming.Update
for addr := range w.addrs {
if _, ok := addrs[addr]; !ok {
updates = append(updates, &naming.Update{Op: naming.Delete, Addr: addr})
}
}
for addr := range addrs {
if _, ok := w.addrs[addr]; !ok {
updates = append(updates, &naming.Update{Op: naming.Add, Addr: addr})
}
}
if len(updates) != 0 {
w.addrs = addrs
return updates, nil
}
}
}
func (w *consulWatcher) Close() {
// nothing to do
}
參考鏈接
gRPC Name Resolution: https://github.com/grpc/grpc/blob/master/doc/naming.md
Load Balancing in gRPC: https://github.com/grpc/grpc/blob/master/doc/load-balancing.md
dns_resolver: https://github.com/grpc/grpc-go/blob/30fb59a4304034ce78ff68e21bd25776b1d79488/naming/dns_resolver.go
代碼地址: https://github.com/hatlonely/hellogolang/blob/master/sample/addservice/cmd/client/main.go
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Go語言實(shí)現(xiàn)廣播式并發(fā)聊天服務(wù)器
本文主要介紹了Go語言實(shí)現(xiàn)廣播式并發(fā)聊天服務(wù)器,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-08-08
GoFrame通用類型變量gvar與interface基本使用對比
這篇文章主要為大家介紹了GoFrame通用類型變量gvar與interface基本使用對比,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
Golang Gin框架實(shí)現(xiàn)多種數(shù)據(jù)格式返回結(jié)果詳解
這篇文章主要介紹了Golang Gin框架實(shí)現(xiàn)多種數(shù)據(jù)格式返回結(jié)果,我們都知道,一個完整的請求包含請求和處理請求以及結(jié)果返回三個步驟,在服務(wù)器端對請求處理完成以后,會將結(jié)果返回給客戶端,在gin框架中,支持返回多種請求數(shù)據(jù)格式,下面我們一起來看看2023-05-05
詳解如何使用unsafe標(biāo)準(zhǔn)庫突破Golang中的類型限制
在使用c語言編程時,常常因?yàn)轭愋偷膯栴}大傷腦筋,而,golang提供了一些方式用于喜歡hack的用戶,下面我們就來講講如何使用unsafe標(biāo)準(zhǔn)庫突破Golang中的類型限制吧2024-03-03
源碼解析gtoken替換jwt實(shí)現(xiàn)sso登錄
這篇文章主要為大家介紹了源碼解析gtoken替換jwt實(shí)現(xiàn)sso登錄的示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
利用Go語言實(shí)現(xiàn)簡單Ping過程的方法
相信利用各種語言實(shí)現(xiàn)Ping已經(jīng)是大家喜聞樂見的事情了,網(wǎng)絡(luò)上利用Golang實(shí)現(xiàn)Ping已經(jīng)有比較詳細(xì)的代碼示例,但大多是僅僅是實(shí)現(xiàn)了Request過程,而對Response的回顯內(nèi)容并沒有做接收。而Ping程序不僅僅是發(fā)送一個ICMP,更重要的是如何接收并進(jìn)行統(tǒng)計。2016-09-09

