go 使用protobuffer proto3

回眸只為那壹抹淺笑 提交于 2019-12-05 00:28:51

1、获取 Protobuf 编译器 protoc,跟C++通用的;可以在git下载到二进制文件


2、获取 goprotobuf 提供的 Protobuf 编译器插件 protoc-gen-go

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

将protoc-gen-go二进制执行程序所在的目录加入到环境变量,或者直接将二进制文件拷贝到 protoc 所在的目录


3.获取 goprotobuf 提供的支持库,包含诸如编码(marshaling)、解码(unmarshaling)等功能

go get github.com/golang/protobuf/proto


4、测试协议 msg.proto:

syntax = "proto3";
package Im;   


message helloworld   
{   
     int32     id = 1;  // ID     
     string    str = 2;  // str    
     int32     opt = 3;  //optional field   
}

5、协议文件得到go文件

protoc --go_out=. msg.proto


msg.pb.go 的内容如下(可以看出proto2中原有的每个字段读取的方法去掉了,现在proto3是直接访问):

// Code generated by protoc-gen-go.
// source: msg.proto
// DO NOT EDIT!

/*
Package Im is a generated protocol buffer package.

It is generated from these files:
	msg.proto

It has these top-level messages:
	Helloworld
*/
package Im

import proto "github.com/golang/protobuf/proto"
import fmt "fmt"
import math "math"

// Reference imports to suppress errors if they are not otherwise used.
var _ = proto.Marshal
var _ = fmt.Errorf
var _ = math.Inf

// This is a compile-time assertion to ensure that this generated file
// is compatible with the proto package it is being compiled against.
// A compilation error at this line likely means your copy of the
// proto package needs to be updated.
const _ = proto.ProtoPackageIsVersion2 // please upgrade the proto package

type Helloworld struct {
	Id  int32  `protobuf:"varint,1,opt,name=id" json:"id,omitempty"`
	Str string `protobuf:"bytes,2,opt,name=str" json:"str,omitempty"`
	Opt int32  `protobuf:"varint,3,opt,name=opt" json:"opt,omitempty"`
}

func (m *Helloworld) Reset()                    { *m = Helloworld{} }
func (m *Helloworld) String() string            { return proto.CompactTextString(m) }
func (*Helloworld) ProtoMessage()               {}
func (*Helloworld) Descriptor() ([]byte, []int) { return fileDescriptor0, []int{0} }

func init() {
	proto.RegisterType((*Helloworld)(nil), "Im.helloworld")
}

func init() { proto.RegisterFile("msg.proto", fileDescriptor0) }

var fileDescriptor0 = []byte{
	// 92 bytes of a gzipped FileDescriptorProto
	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x09, 0x6e, 0x88, 0x02, 0xff, 0xe2, 0xe2, 0xcc, 0x2d, 0x4e, 0xd7,
	0x2b, 0x28, 0xca, 0x2f, 0xc9, 0x17, 0x62, 0xf2, 0xcc, 0x55, 0x32, 0xe2, 0xe2, 0xca, 0x48, 0xcd,
	0xc9, 0xc9, 0x2f, 0xcf, 0x2f, 0xca, 0x49, 0x11, 0xe2, 0xe2, 0x62, 0xca, 0x4c, 0x91, 0x60, 0x54,
	0x60, 0xd4, 0x60, 0x15, 0xe2, 0xe6, 0x62, 0x2e, 0x2e, 0x29, 0x92, 0x60, 0x52, 0x60, 0xd4, 0xe0,
	0x04, 0x71, 0xf2, 0x0b, 0x4a, 0x24, 0x98, 0x41, 0x32, 0x49, 0x6c, 0x60, 0xed, 0xc6, 0x80, 0x00,
	0x00, 0x00, 0xff, 0xff, 0xc8, 0x9e, 0x4a, 0x79, 0x4b, 0x00, 0x00, 0x00,
}



6、文件入库

方案A,入当前工程,使用 import  "./Im"引用:

当前工程目录下新建一个Im的目录(因为协议package 名叫Im)

将生成的msg.pb.go 文件拷贝到此Im目录中


方案B,入go的pkg库,使用  import  "Im"引用:

到go的pkg/src目录下新建一个Im的目录(因为协议package 名叫Im)

将生成的msg.pb.go 文件拷贝到此Im目录中


7、

go测试代码 mainpb.go:

package main

import (
    "log"
    // 辅助库
    "github.com/golang/protobuf/proto"
    // x.pb.go 的路径
    "./Im"
)

func main() {
    // 创建一个消息 
     test := &Im.Helloworld{
        // 使用辅助函数设置域的值
        Str: "hello!" ,
      //  Id:  321,
        Opt: 1234,
    }

    test.Id = 3244

    // 进行编码
    data, err := proto.Marshal(test)
    if err != nil {
        log.Fatal("marshaling error: ", err)
    }

    // 进行解码
    newTest := &Im.Helloworld{}
    err = proto.Unmarshal(data, newTest)
    if err != nil {
        log.Fatal("unmarshaling error: ", err)
    }

    log.Printf("id:%d;opt:%d;str:%s;",newTest.Id,newTest.Opt,newTest.Str)

    // 测试结果
    if test.String() != newTest.String() {
        log.Fatalf("data mismatch %q != %q", test.String(), newTest.String())
    }
}


go run mainpb.go

运行结果:
2016/09/08 16:27:52 id:321;opt:1234;str:hello!;


易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!