GO语言:Protobuf协议Go语言实践 Davie

我会带着你远行 2023-10-06 15:16 93阅读 0赞

Go语言中有对应的实现Protobuf协议的库,Github地址:https://github.com/golang/protobuf

环境准备

使用Go语言的Protobuf库之前,需要相应的环境准备:

1、安装protobuf编译器。 可以在如下地址:https://github.com/protocolbuffers/protobuf/releases选择适合自己系统的Proto编译器程序进行下载并解压,如图:

400e64177a2fdbb3078dafdd10c8a676.png

2、配置环境变量 protoc编译器正常运行需要进行环境变量配置,将protocke执行文件所在目录添加到当前系统的环境变量中。windows系统下可以直接在Path目录中进行添加;macOS系统下可以将protoc可执行文件拷贝至/usr/local/include目录下。具体的对应的系统的环境变量配置可以阅读解压后与bin目录同级的readme.txt的文件内容。

安装

通过如下命令安装protoc-gen-go库:

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

fdca74ac4d834280adaa10ee69a81968.png

安装完成以后,protoc-gen-go*可执行文件在本地环境GOPATH/bin目录下,如下图所示:

3d12301927b9488fb617eb89aa55c875.png

Protobuf 协议语法

  • Protobuf 协议的格式 Protobuf协议规定:使用该协议进行数据序列化和反序列化操作时,首先定义传输数据的格式,并命名为以“.proto”为扩展名的消息定义文件。
  • message 定义一个消息 先来看一个非常简单的例子。假设想定义一个“订单”的消息格式,每一个“订单”都含有一个订单号ID、订单金额Num、订单时间TimeStamp字段。可以采用如下的方式来定义消息类型的.proto文件:

  • 指定字段类型:在proto协议中,字段的类型包括字符串(string)、整形(int32、int64…)、枚举(enum)等数据类型

  • 分配标识符:在消息字段中,每个字段都有唯一的一个标识符。最小的标识号可以从1开始,最大到536870911。不可以使用其中的[19000-19999]的标识号, Protobuf协议实现中对这些进行了预留。如果非要在.proto文件中使用这些预留标识号,编译时就会报警。
  • 指定字段规则:字段的修饰符包含三种类型,分别是:

    • required:一个格式良好的消息一定要含有1个这种字段。表示该值是必须要设置的;
    • optional:消息格式中该字段可以有0个或1个值(不超过1个)。
    • repeated:在一个格式良好的消息中,这种字段可以重复任意多次(包括0次)。重复的值的顺序会被保留。表示该值可以重复,相当于Go中的slice。

【注意:】使用required弊多于利;在实际开发中更应该使用optional和repeated而不是required。

  • 添加更多消息类型 在同一个.proto文件中,可以定义多个消息类型。多个消息类型分开定义即可。

使用 Protobuf 的步骤

1、创建扩展名为.proto的文件,并编写代码。比如创建person.proto文件,内容如下:

  1. syntax = "proto2";
  2. package example;
  3. message Person {
  4. required string Name = 1;
  5. required int32 Age = 2;
  6. required string From = 3;
  7. }

2、编译.proto文件,生成Go语言文件。执行如下命令:

  1. protoc --go_out = . test.proto

执行 protoc —go_out=. test.proto 生成对应的 person.pb.go 文件。并构建对应的example目录,存放生成的person.pb.go文件。

419a0418bba04ad7b6e314ecf5de04b0.png

3、在程序中使用Protobuf 在程序中有如下代码:

  1. package main
  2. import (
  3. "fmt"
  4. "ProtocDemo/example"
  5. "github.com/golang/protobuf/proto"
  6. "os"
  7. )
  8. func main() {
  9. fmt.Println("Hello World. \n")
  10. msg_test := &example.Person{
  11. Name: proto.String("Davie"),
  12. Age: proto.Int(18),
  13. From: proto.String("China"),
  14. }
  15. //序列化
  16. msgDataEncoding, err := proto.Marshal(msg_test)
  17. if err != nil {
  18. panic(err.Error())
  19. return
  20. }
  21. msgEntity := example.Person{}
  22. err = proto.Unmarshal(msgDataEncoding, &msgEntity)
  23. if err != nil {
  24. fmt.Println(err.Error())
  25. os.Exit(1)
  26. return
  27. }
  28. fmt.Printf("姓名:%s\n\n", msgEntity.GetName())
  29. fmt.Printf("年龄:%d\n\n", msgEntity.GetAge())
  30. fmt.Printf("国籍:%s\n\n", msgEntity.GetFrom())
  31. }

4、执行程序

9f8fa7be75e84a3b8f426a1e717514bc.png

发表评论

表情:
评论列表 (有 0 条评论,93人围观)

还没有评论,来说两句吧...

相关阅读

    相关 GO语言Protobuf简介

    RPC 通信 对于单独部署,独立运行的微服务实例而言,在业务需要时,需要与其他服务进行通信,这种通信方式是进程之间的通讯方式(inter-process communic