it编程 > 编程语言 > rust

protobuf简介及使用流程

17人参与 2025-02-26 rust

1. protobuf是什么

  protobuf(全称protocol buffer)是数据结构序列化和反序列化框架,它具有以下特点:

2. protobuf使⽤流程介绍

3. protobuf快速上手

  我们以⼀个简单通讯录的实现来驱动对protobuf的学习。在通讯录demo中,我们将实现:

  通过通讯录demo,我们能快速的了解protobuf的使⽤流程。

3.1 创建 .proto ⽂件

  我们为通讯录 demo 新建⽂件: contacts.proto

3.2 添加注释

  向⽂件添加注释,可使⽤ // 或者 /* … */

3.3 具体编写

  指定 proto3 语法:
  protocol buffers 语⾔版本3,简称 proto3,是 .proto ⽂件最新的语法版本。proto3 简化了 protocolbuffers 语⾔,既易于使⽤,⼜可以在更⼴泛的编程语⾔中使⽤。它允许你使⽤ java,c++,python等多种语⾔⽣成 protocol buffer 代码。在 .proto ⽂件中,要使⽤ syntax = “proto3”; 来指定⽂件语法为 proto3,并且必须写在除去注释内容的第⼀⾏。 如果没有指定,编译器会使⽤proto2语法。
  在通讯录 demo 的 contacts.proto ⽂件中,可以为⽂件指定 proto3 语法,内容如下:

syntax = "proto3";

   package 声明符:
   package 是⼀个可选的声明符,能表⽰ .proto ⽂件的命名空间,在项⽬中要有唯⼀性。它的作⽤是为了避免我们定义的消息出现冲突。
   在通讯录 demo 的 contacts.proto ⽂件中,可以声明其命名空间,内容如下:

package contacts;

   定义消息(message):
  消息(message): 要定义的结构化对象,我们可以给这个结构化对象中定义其对应的属性内容。在网络传输中,我们需要为传输双⽅定制协议。定制协议说⽩了就是定义结构体或者结构化数据,⽐如,tcp,udp 报⽂就是结构化的。再⽐如将数据持久化存储到数据库时,会将⼀系列元数据统⼀⽤对象组织起来,再进⾏存储。protobuf 就是以 message 的方式来⽀持我们定制协议字段,后期帮助我们形成类和⽅法来使用。
  在通讯录 demo 中我们就需要为联系人定义⼀个 message:

.proto ⽂件中定义⼀个消息类型的格式为:

message 消息类型名{
}
消息类型命名规范:使⽤驼峰命名法,⾸字⺟⼤写。

为 contacts.proto(通讯录 demo)新增联系⼈message

syntax = "proto3";
package contacts;
// 定义联系⼈消息
message peopleinfo {
}

  定义消息字段:
  在 message 中我们可以定义其属性字段,字段定义格式为:字段类型 字段名 = 字段唯⼀编号;

  样例:

// 声明语法版本
syntax = "proto3";
// 声明代码的命名空间
package contacts;
//结构化对象的描述
message peopleinfo{
    // 各个字段描述:   字段类型  字段名 = 字段唯一编号
    string name = 1;
    int32 age = 2;
}

  注:这⾥还要特别讲解⼀下字段唯⼀编号的范围:1 ~ 536,870,911 (2^29 - 1) ,其中 19000 ~ 19999 不可⽤。
  19000 ~ 19999 不可⽤是因为:在 protobuf 协议的实现中,对这些数进⾏了预留。如果⾮要在.proto⽂件中使⽤这些预留标识号,例如将 name 字段的编号设置为19000,编译时就会报警。
  值得⼀提的是,范围为 1 ~ 15 的字段编号需要⼀个字节进⾏编码, 16 ~ 2047 内的数字需要两个字节进⾏编码。编码后的字节不仅只包含了编号,还包含了字段类型。所以 1 ~ 15 要⽤来标记出现⾮常频繁的字段,要为将来有可能添加的、频繁出现的字段预留⼀些出来。

3.4 编译 contacts.proto 文件

  编译命令⾏格式为:

protoc [--proto_path=import_path] --cpp_out=dst_dir path/to/file.proto
protoc             是 protocol buffer 提供的命令⾏编译⼯具。
--proto_path       指定被编译的.proto⽂件所在⽬录,可多次指定。可简写成 -i import_path 。如不指
                   定该参数,则在当前⽬录进⾏搜索。当某个.proto ⽂件 import 其他.proto ⽂件时,
                   或需要编译的 .proto ⽂件不在当前⽬录下,这时就要⽤-i来指定搜索⽬录。
--cpp_out=         指编译后的⽂件为 c++ ⽂件。
out_dir            编译后⽣成⽂件的⽬标路径。
path/to/file.proto 要编译的.proto⽂件

  编译 contacts.proto ⽂件命令如下:

protoc --cpp_out=. contacts.proto

  编译 contacts.proto ⽂件后,会⽣成所选择语⾔的代码,我们选择的是c++,所以编译后⽣成了两个文件: contacts.pb.h contacts.pb.cc 。
  对于编译⽣成的 c++ 代码,包含了以下内容 :

  contacts.pb.h 部分代码展示:

class peopleinfo final : public ::protobuf_namespace_id::message {
public:
	 using ::protobuf_namespace_id::message::copyfrom;
	 void copyfrom(const peopleinfo& from);
	 using ::protobuf_namespace_id::message::mergefrom;
	 void mergefrom( const peopleinfo& from) {
	 peopleinfo::mergeimpl(*this, from);
 }
 static ::protobuf_namespace_id::stringpiece fullmessagename() {
	 return "peopleinfo";
 }
	 // string name = 1;
	 void clear_name();
	 const std::string& name() const;
	 template <typename argt0 = const std::string&, typename... argt>
	 void set_name(argt0&& arg0, argt... args);
	 std::string* mutable_name();
	 protobuf_nodiscard std::string* release_name();
	 void set_allocated_name(std::string* name);
	 // int32 age = 2;
	 void clear_age();
	 int32_t age() const;
	 void set_age(int32_t value);
};

  上述的例⼦中:

  contacts.pb.cc 中的代码就是对类声明⽅法的⼀些实现,在这⾥就不展开了。
  到这⾥有人可能就有疑惑了,那之前提到的序列化和反序列化⽅法在哪⾥呢?在消息类的⽗类messagelite 中,提供了读写消息实例的方法,包括序列化⽅法和反序列化⽅法。

class messagelite {
public:
	//序列化:
	bool serializetoostream(ostream* output) const; // 将序列化后数据写⼊⽂件
	流
	bool serializetoarray(void *data, int size) const;
	bool serializetostring(string* output) const;
	//反序列化:
	bool parsefromistream(istream* input); // 从流中读取数据,再进⾏反序列化
	动作
	bool parsefromarray(const void* data, int size);
	bool parsefromstring(const string& data);
};

  注意:

  序列化与反序列化的使用:
  创建⼀个测试⽂件 test.cc,⽅法中我们实现:

#include <iostream>
#include "contacts.pb.h"
using namespace std;
int main()
{
    string people_str;
    {
        contacts::peopleinfo people;
        people.set_age(20);
        people.set_name("忘忧");
        if(!people.serializetostring(&people_str))
        {
            cout << "序列化联系人失败" <<endl;
        }
        cout << "序列化之后的 people_str: " << people_str << endl;
        // 反序列化
        {
            contacts::peopleinfo people;
            if(!people.parsefromstring(people_str))
            {
                cout << "反序列化联系人失败" <<endl;
            }
            cout << "parse age: " << people.age() << endl;
            cout << "parse name: " << people.name() << endl;
        }
    }
    return 0;
}

代码书写完成后,编译 test.cc,生成可执行程序:

g++ test.cc contacts.pb.cc -o test -std=c++11 -lprotobuf

执⾏可执⾏程序,可以看⻅ people 经过序列化和反序列化后的结果:

  由于 protobuf 是把联系⼈对象序列化成了⼆进制序列,这⾥⽤ string 来作为接收⼆进制序列的容器。所以在终端打印的时候会有换⾏等⼀些乱码显⽰。另外相对于 xml 和 json 来说,因为pb被编码成⼆进制,破解成本增⼤,protobuf 编码是相对安全的。

到此这篇关于protobuf简介及使用流程的文章就介绍到这了,更多相关protobuf使用内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)
打赏 微信扫一扫 微信扫一扫

您想发表意见!!点此发布评论

推荐阅读

Rust之Rhai脚本编程的示例

02-26

Rust中的模块系统之控制作用域与私有性详解

02-26

Rust中的方法与关联函数使用解读

02-26

Rust中的注释使用解读

02-26

Rust中的Trait与Trait Bounds详解

02-26

使用环境变量实现Rust程序中的不区分大小写搜索方式

02-26

猜你喜欢

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

发表评论