• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    golang grpc 负载均衡的方法

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

    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.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
    }
    

    参考链接

    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

    以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

    您可能感兴趣的文章:
    • Golang实现四种负载均衡的算法(随机,轮询等)
    • 使用Golang实现加权负载均衡算法的实现代码
    上一篇:Go语言包管理工具dep的安装与使用
    下一篇:Go语言中的上下文取消操作详解
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    golang grpc 负载均衡的方法 golang,grpc,负载,均衡,的,