go的grpc实现

  gRPC是Google的RPC框架,开源、高性能、跨语言,基于HTTP/2通讯协议和Protocol Buffer 3数据序列化协议。
过程如下图所示:

  调用的双方可以使用完全不同的两种语言来实现,分别实现client端和server端,按照约定的protobuf协议进行交互。client端会保存与server端的长连接对象或叫存根,通过这个存根可以直接调用服务端的方法。而服务端则实现了proto中指定的服务接口。

proto的安装

要编译proto文件生成go代码需要两个工具

  • protoc :用于编译(其他语言只需要protoc足以)
  • protoc-gen-go : 用于生成go语言的文件(go语言专用插件)

1、protoc-gen-go下载:

go get github.com/golang/protobuf/protoc-gen-go

会直接生成protoc-gen-go 二进制文件到GOPATH的bin目录。
2、protoc下载:https://github.com/google/protobuf/releases
找到对应平台win/linux/mac的下载包,复制二进制文件protoc到GOPATH的bin目录。

定义proto文件

syntax = "proto3";

package protos;

// 要获取的数据结构
message User{
    int32 id = 1;
    string name = 2;
}

// 请求数据结构
message UserReq{
    int32 id = 1;
}

// 定义服务,关键字'service',方法关键字'rpc'
service IUserService {
    // 单一请求应答,一对一
    rpc Get (UserReq) returns (User);
    // 服务端流式应答,一对多,可用于下载
    rpc GetList (UserReq) returns (stream User);
    // 客户端流式请求,多对一,可用于上传
    rpc WaitGet(stream UserReq) returns (User);
    // 双向流式请求应答,支持HTTP/2.0
    rpc LoopGet(stream UserReq) returns (stream User);
}

编译proto生成go文件

这里要使用grpc插件选项:

protoc --go_out=plugins=grpc:. 路径/*.proto

会在同目录下生成.pb.go文件。如果不识别protoc,则说明GOPATH的bin目录不在环境变量中,需要设置。

安装grpc包

go get google.golang.org/grpc

实现grpc的server端

1、接口实现,即对外提供的服务API。

package service

import (
    "fmt"
    "io"
    "log"
    "protos"
    "strconv"

    "golang.org/x/net/context"
)

//对外提供的工厂函数
func NewUserService() *UserService {
    return &UserService{}
}

//**************************************************************
// 接口实现,接口定义是在proto生成的.pb.go文件中
//**************************************************************

// 接口实现对象,属性成员根据而业务自定义
type UserService struct {
}

// Get接口方法实现
func (this *UserService) Get(ctx context.Context,req *protos.UserReq) (*protos.User,error) {
    return &protos.User{Id: 1,Name: "shuai"},nil
}

// GetList接口方法实现
func (this *UserService) GetList(req *protos.UserReq,stream protos.IUserService_GetListServer) error {
    fmt.Println(*req)
    // 流式返回多条数据
    for i := 0; i < 5; i++ {
        stream.Send(&protos.User{Id: int32(i),Name: "我是" + strconv.Itoa(i)})
    }
    return nil
}

// WaitGet接口方法实现
func (this *UserService) WaitGet(reqStream protos.IUserService_WaitGetServer) error {
    for { // 接收流式请求并返回单一对象
        userReq,err := reqStream.Recv()
        if err != io.EOF {
            fmt.Println("流请求~",*userReq)
        } else {
            return reqStream.SendAndClose(&protos.User{Id: 100,Name: "shuai"})
        }
    }
}

//双向流:请求流和响应流异步
func (this *UserService) LoopGet(reqStream protos.IUserService_LoopGetServer) error {
    for {
        userReq,err := reqStream.Recv()
        if err == io.EOF { //请求结束
            return nil
        }
        if err != nil {
            return err
        }
        if err = reqStream.Send(&protos.User{Id: userReq.Id,Name: "shuai"}); err != nil {
            return err
        }
    }
}

2、启动服务

package main

import (
    "flag"
    "fmt"
    "net"
    "proj/service" // 实现了服务接口的包service
    "protos" // 此为自定义的protos包,存放的是.proto文件和对应的.pb.go文件

    "google.golang.org/grpc"
)

var (
    // 命令行参数-host,默认服务监听端口在9000
    addr = flag.String("host","127.0.0.1:9000","")
)

func main() {
    // 开启服务监听
    lis,err := net.Listen("tcp",*addr)
    if err != nil {
        fmt.Println("listen error!")
        return
    }
    // 创建一个grpc服务
    grpcServer := grpc.NewServer()
    // 重点:向grpc服务中注册一个api服务,这里是UserService,处理相关请求
    protos.RegisterIUserServiceServer(grpcServer,service.NewUserService())
    // 可以添加多个api
    // TODO...

    // 启动grpc服务
    grpcServer.Serve(lis)
}
go run main.go //或指定-host "localhost:9001"

实现grpc的client端

package main

import (
    "flag"
    "fmt"
    "io"
    "log"
    "protos" // 此为自定义的protos包,存放的是.proto文件和对应的.pb.go文件

    "golang.org/x/net/context"

    "google.golang.org/grpc"
)

var (
    // 命令行参数-host,默认地址本机9000端口
    addr = flag.String("host","")
    // UserService服务存根,可直接调用服务方法
    userCli protos.IUserServiceClient
)

func main() {
    // 创建grpc的服务连接
    conn,err := grpc.Dial(*addr)
    if err != nil {
        log.Fatal("failed to connect : ",err)
    }
    // 存根
    userCli = protos.NewIUserServiceClient(conn)

    // 测试前三种数据传递方式,第四种省略(二三结合)
    TestGet()
    TestGetList()
    TestWaitGet()
}

func TestGet() {
    // 测试调用方法Get,返回user对象
    user,err := userCli.Get(context.Background(),&protos.UserReq{Id: 1})
    if err != nil {
        fmt.Printf("Get connect failed :%v",err)
        return
    }
    log.Println("Get响应数据:",*user)
}

func TestGetList() {
    // 测试调用方法GetList,返回一个Stream流,循环获取多个user对象
    recvStream,err := userCli.GetList(context.Background(),&protos.UserReq{Id: 1})
    if err != nil {
        fmt.Printf("GetList connect failed :%v",err)
        return
    }
    for {
        user,err := recvStream.Recv()
        if err == io.EOF {
            break
        }
        log.Println("GetList获取的一条响应数据:",*user)
    }
}

func TestWaitGet() {
    // 测试调用方法WaitGet,传入多条请求数据,返回一个user对象
    sendStream,err := userCli.WaitGet(context.Background())
    if err != nil {
        fmt.Printf("WaitGet connect failed :%v",err)
        return
    }
    for i := 0; i < 5; i++ { // 一次传入5条请求数据
        if err = sendStream.Send(&protos.UserReq{Id: int32(i)}); err != nil {
            fmt.Printf("WaitGet send failed :%v",err)
            return
        }
    }
    // 服务端接受全部请求数据后,返回一个user对象
    user,err := sendStream.CloseAndRecv()
    if err != nil {
        fmt.Printf("WaitGet recv failed :%v",err)
        return
    }
    log.Println("WaitGet响应数据:",*user)
}

启动client端:

go run main.go

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


gRPC 前言 为什么使用gRPC 传输协议 传输效率 性能消耗 gRPC入门 gRPC流 证书认证 使用根证书 gRPC实现token认证 和Web服务共存 验证器 REST接口 grpcurl工具
参考文章: 1.&#160;https://www.cnblogs.com/kaixinyufeng/p/9651513.html 2.&#160;http://jia-shun.cn/2018/08
今天给大家翻译一篇由ASP.NET首席开发工程师 &quot;James Newton King&quot; 前几天发表的一篇博客,文中带来了一个实验性的产品gRPC Web。大家可以点击文末的讨论帖
上一篇文章我带着大家体验了一把《 &quot;ASP.NET Core 3.0 上的gRPC服务模板初体验(多图)&quot; 》,如果有兴趣的可以点击链接进行查看,相信跟着做的你,也是可以跑起来的。
早就听说ASP.NET Core 3.0中引入了gRPC的服务模板,正好趁着家里电脑刚做了新系统,然后装了VS2019的功夫来体验一把。同时记录体验的过程。如果你也想按照本文的步骤体验的话,那你得先安
这篇笔记主要是记录学习历程而不是怎么用~,以及protobuffers 和 gprc 各种文档的地址,等过上大半年后通过这篇笔记帮助自己快速重新掌握这个技术点 一、Protocolbuffers 关于
最近GRPC很火,感觉整RPC不用GRPC都快跟不上时髦了。 gRPC设计 gRPC是一种与语言无关的高性能远程过程调用 (RPC) 框架。刚好需要使用一个的RPC应用系统,自然而然就盯上了它,但是它
   gRPC是google开源提供的一个RPC软件框架,它的特点是极大简化了传统RPC的开发流程和代码量,使用户可以免除许多陷阱并聚焦于实际应用逻辑中。作为一种google的最新RPC解决方案,gRPC具备了以下这些强项: 1、gRPC在HTTP/2协议上用protobuf取代了json实现了最佳效率 2、用IDL(Interface Definition Language),一种简单的描述语言
  接着上期讨论的gRPC unary服务我们跟着介绍gRPC streaming,包括: Server-Streaming, Client-Streaming及Bidirectional-Streaming。我们首先在.proto文件里用IDL描述Server-Streaming服务: /* * responding stream of increment results */ servi
我已经设法通过GRPC使用流媒体模式的服务帐户为我的 Android应用程序运行Google Cloud Speech.但是,根据我所读到的内容,出于安全原因,我不应该在其中部署具有这些凭据的Android应用程序(当前存储为资源中的JSON文件).正确的是创建一个API密钥,如下所述: https://cloud.google.com/speech/docs/common/auth 这允许我限制
  安装protobuf go get -u github.com/golang/protobuf/proto go get -u github.com/golang/protobuf/protoc-gen-go 此时会生成protoc-gen-go,protoc一般是获取已经编译好的可执行文件(https://github.com/google/protobuf/releases)   安装gR
一、grpc安装 将 https://github.com/google/go-genproto 放到 $GOPATH/src/google.golang.org/genproto 将 https://github.com/grpc/grpc-go 放到 $GOPATH/src/google.golang.org/grpc 将 https://github.com/golang/t
参考URL: https://segmentfault.com/a/1190000015220713?utm_source=channel-hottest gRPC 是一个高性能、开源和通用的 RPC 框架,面向移动和 HTTP/2 设计。目前提供 C、Java 和 Go 语言版本,分别是:grpc, grpc-java, grpc-go. 其中 C 版本支持 C, C++, Node.js, P
我试图在电子应用程序中要求grpc,但我收到以下错误: Error: dlopen(/srv/node_modules/grpc/src/node/extension_binary/grpc_node.node, 1): Symbol not found: _GENERAL_NAME_free Referenced from: /srv/node_modules/grpc/src/node/e
我试图调用GRPC端点,但我想提供客户身份验证标头.我在哪里指定这个? var client = new proto.Publisher('127.0.0.1:50051', grpc.credentials.createInsecure()); var customHeader = { 'authorization': 'secret' } client.publish(d
我正在尝试创建一个 java grpc客户端来与go中的服务器通信.我是grpc的新手所以遵循本教程 gRPC Java Tutorial.在这些示例中,它们指的是阻塞和非阻塞存根,它们似乎是从 github的其他地方导入的. import io.grpc.examples.routeguide.RouteGuideGrpc.RouteGuideBlockingStub; import io.gr
我正在尝试做类似下面的事情(即使用流式grpc调用从客户端向服务器发送数据).代码参考取自官方网站上给出的grpc示例,用于解释目的: 客户端代码: ClientContext context; context.AddMetadata("authorization", "abcd"); context.set_deadline(...); std::unique_ptr<ClientWriter
什么是gRPC gRPC是google开源的一个高性能、跨语言的RPC框架,基于HTTP2协议,采用ProtoBuf 定义的IDL。 gRPC 的主要优点是: 现代高性能轻量级 RPC 框架。 协定优先 API 开发,默认使用协议缓冲区,允许与语言无关的实现。 可用于多种语言的工具,以生成强类型服务器和客户端。 支持客户端、服务器和双向流式处理调用。 使用 Protobuf 二进制序列化减少对网络
一.简介 gRPC 是一个由Google开源的,跨语言的,高性能的远程过程调用(RPC)框架。 gRPC使客户端和服务端应用程序可以透明地进行通信,并简化了连接系统的构建。它使用HTTP/2作为通信协议,使用 Protocol Buffers 作为序列化协议。 它的主要优点: 现代高性能轻量级 RPC 框架。 约定优先的 API 开发,默认使用 Protocol Buffers 作为描述语言,允许
目录 ASP.NET Core 3.0 使用gRPC ASP.NET Core 3.0 gRPC 双向流 ASP.NET Core 3.0 gRPC 认证授权 一.前言 在前一文 《ASP.NET Core 3.0 使用gRPC》中有提到 gRPC 支持双向流调用,支持实时推送消息,这也是 gRPC的一大特点,且 gRPC 在对双向流的控制支持上也是非常强大的。 二. 什么是 gRPC 流 gRP