分享
  1. 首页
  2. 文章

golang grpc 负载均衡

hatlonely · · 2555 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

微服务架构里面,每个服务都会有很多节点,如果流量分配不均匀,会造成资源的浪费,甚至将一些机器压垮,这个时候就需要负载均衡,最简单的一种策略就是轮询,顺序依次选择不同的节点访问

grpc 在客户端提供了负载均衡的实现,并提供了服务地址解析和更新的接口(默认提供了 DNS 域名解析的支持),方便不同服务的集成

使用示例

conn, err := grpc.Dial(
 "",
 grpc.WithInsecure(),
 // 负载均衡,使用 consul 作服务发现
 grpc.WithBalancer(grpc.RoundRobin(grpclb.NewConsulResolver(
 "127.0.0.1:8500", "grpc.health.v1.add",
 ))),
)

创建连接的时候可以使用 WithBalancer 选项来指定负载均衡策略,这里使用 RoundRobin 算法,其实就是轮询策略

与 consul 的集成

有了负载均衡策略,还需要一个地址解析和更新策略,可以使用 DNS 服务来实现,但如果我们使用 consul 来做服务的注册和发现,可以通过实现 naming.Resolvernaming.Watcher 接口来支持

  • naming.Resolver: 实现地址解析
  • naming.Watcher: 实现节点的变更,添加或者删除
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, // 同步点,这个调用将一直阻塞,直到有新的更新
 })
 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
}

参考链接

转载请注明出处
本文链接:http://www.hatlonely.com/2018/06/23/golang-grpc-%E8%B4%9F%E8%BD%BD%E5%9D%87%E8%A1%A1/


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:hatlonely

查看原文:golang grpc 负载均衡

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
2555 次点击
被以下专栏收入,发现更多相似内容
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏