Protobuf 二进制文件学习及解析

0. 简介

protobuf也叫protocol buffer是google 的一种数据交换的格式,它独立于语言,独立于平台。google 提供了多种语言的实现:java、c#、c++、go 和 python,每一种实现都包含了相应语言的编译器以及库文件。

由于它是一种二进制的格式,比使用 xml 、json进行数据交换快许多。可以把它用于分布式应用之间的数据通信或者异构环境下的数据交换。作为一种效率和兼容性都很优秀的二进制数据传输格式,可以用于诸如网络传输、配置文件、数据存储等诸多领域。

更多详情请看:https://developers.google.com/protocol-buffers/docs/overview

1. 基础语法

1 字段限制

required: 必须赋值的字符
optional: 可有可无的字段,可以使用[default = xxx]配置默认值
repeated: 可重复变长字段,类似数组

2 tag

每个字段都有独一无二的tag
tag 1-15是字节编码,16-2047使用2字节编码,所以1-15给频繁使用的字段

3 类型

在这里插入图片描述

系统默认值:
string默认为空字符串;
bool默认为false;
数值默认为0;
enum默认为第一个元素

4 解析与序列化

每个message都包含如下方法,用于解析和序列化,注意目标是字节形式,非文本。
bool SerializeToString(string* output) const: 将message序列化成二进制保存在output中,注意保存的是二进制,不是文本;仅仅是string作为容器。
bool ParseFromString(const string& data): 从给定的二进制数值中解析成message
bool SerializeToOstream(ostream* output) const: 序列化到ostream中
bool ParseFromIstream(istream* input): 从istream中解析出message

3. 讲解举例

建立.proto文件
syntax = "proto3";//指定版本信息,不指定会报错message Person //message为关键字,作用为定义一种消息类型{    string name = 1;    //姓名    int32 id = 2;       //id    string email = 3;   //邮件}message AddressBook{    repeated Person people = 1;}
字段API

一般我们会经常使用protoc来自动生成:

protoc -I=$SRC_DIR --cpp_out=$DST_DIR $SRC_DIR/addressbook.proto

其中protoc工具地址:protoc

而对于字段修饰符为repeated的字段生成的函数,则稍微有一些不同,如people字段,则编译器会为其产生如下的代码:

int people_size() const;void clear_people();const ::Person& people(int index) const;::Person* mutable_people(int index);::Person* add_people();::google::protobuf::RepeatedPtrField< ::Person >* mutable_people();const ::google::protobuf::RepeatedPtrField< ::Person >& people() const;
测试程序
void set_addressbook(){    AddressBook obj;    Person *p1 = obj.add_people(); //新增加一个Person    p1->set_name("mike");    p1->set_id(1);    p1->set_email("mike@qq.com");    Person *p2 = obj.add_people(); //新增加一个Person    p2->set_name("jiang");    p2->set_id(2);    p2->set_email("jiang@qq.com");    Person *p3 = obj.add_people(); //新增加一个Person    p3->set_name("abc");    p3->set_id(3);    p3->set_email("abc@qq.com");    fstream output("pb.xxx", ios::out | ios::trunc | ios::binary);    bool flag = obj.SerializeToOstream(&output);//序列化    if (!flag)    {        cerr << "Failed to write file." << endl;        return;    }    output.close();//关闭文件}void get_addressbook(){    AddressBook obj;    fstream input("./pb.xxx", ios::in | ios::binary);    obj.ParseFromIstream(&input);  //反序列化    input.close(); //关闭文件    for (int i = 0; i < obj.people_size(); i++)    {        const Person& person = obj.people(i);//取第i个people        cout << "第" << i + 1 << "个信息\n";        cout << "name = " << person.name() << endl;        cout << "id = " << person.id() << endl;        cout << "email = " << person.email() << endl << endl;    }}
运行结果:

在这里插入图片描述

点击https://www.guyuehome.com/36919可查看全文

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/589043.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【图论】【树】 【拓扑排序】2603. 收集树中金币

本文涉及知识点 图论 树 拓扑排序 LeetCode 2603. 收集树中金币 给你一个 n 个节点的无向无根树,节点编号从 0 到 n - 1 。给你整数 n 和一个长度为 n - 1 的二维整数数组 edges ,其中 edges[i] [ai, bi] 表示树中节点 ai 和 bi 之间有一条边。再给你…

Vue 大文件切片上传实现指南包会,含【并发上传切片,断点续传,服务器合并切片,计算文件MD5,上传进度显示,秒传】等功能

Vue 大文件切片上传实现指南 背景 在Web开发中,文件上传是一个常见的功能需求,尤其是当涉及到大文件上传时,为了提高上传的稳定性和效率,文件切片上传技术便显得尤为重要。通过将大文件切分成多个小块(切片&#xff0…

提效提速的快捷回复工具

在数字化交流日益增长的今天,客服工作显得尤为重要。为了提升对话质量和回复速度,同时减少重复劳动,我同事给我介绍了一款快捷回复工具,叫做客服宝聊天助手。我用了几天真心觉得好好用,今天特地分享这个软件给你们&…

【编译lombok问题】已解决:编译突然找不到符号问题-get/set找不到符号

一、场景:编译突然找不到符号 报错信息: 找不到符号 符号:方法getName() 二、原因: 没有使用lombok支持的编译器 三、解决方法: 打开File-Settings,按以下步骤进行设置; 修改:-Djp…

QuartusII联合Modelsim仿真中最好不要将tb文件设置为顶层,以避免compile错误

QuartusII联合Modelsim仿真中最好不要将tb文件设置为顶层,以避免compiler错误 1,QuartusII下的rtl文件、sim文件如下显示。2,将rtl文件中任一个置于顶层,都不会影响sim仿真输出。3,将tb.v文件置于顶层,comp…

windows无法使用hadoop报错:系统找不到路径

在windows下安装hadoop-3.1.4,进行环境变量配置后,打开window命令行窗口测试hadoop命令,报错,如图所示: 方案:由于JAVA_HOME路径有空格导致,可修改hadoop下\etc\hadoop\hadoop_env.cmd文档中set JAVA_HOME以…

VTK| VTK可视化流程+圆锥示例

要想入门vtk,了解vtk的可视化流程是非常有必要的。 VTK可视化流程 VTK可视化流程主要分为数据处理和渲染两个过程,有一张不错的可视化流程图把这个过程理解为一个舞台剧。 VTKVS运行圆锥示例 先来运行一个简单的示例代码来理解VTK运作的过程&#xff…

k8s 基础入门

1.namespace k8s中的namespace和docker中namespace是两码事,可以理解为k8s中的namespace是为了多租户,dockers中的namespace是为了网络、资源等隔离 2.deployment kubectl create #新建 kubectl aply #新建 更新 升级: 滚动升级&#x…

CSS水波纹效果

效果图&#xff1a; 1.创建一个div <div class"point1" click"handlePoint(1)"></div> 2.设置样式 .point1{width: 1rem;height: 1rem;background: #2ce92f;position: absolute;border-radius: 50%;z-index: 999;cursor: pointer;} 3.设置伪…

C++初学者:优雅创建第一个窗口

我想学习C做一些实用的程序&#xff0c;但是我不想在软件界面上花太多的时间&#xff0c;可是每每就是界面影响我的思绪。 今天学习C类的包装知识&#xff0c;终于整出了一个我的界面类&#xff0c;虽然封装水平很弱&#xff0c; 这次就用这个类&#xff0c;写了自己工作上常用…

什么是视频号小店?视频号小店怎么开通,为什么那么多人都在做?

大家好&#xff0c;我是电商花花。 要说现在最火的行业/项目是什么&#xff1f;相信每个人都下意识觉得是直播电商。 没错&#xff0c;自动直播电商火了之后&#xff0c;更是带动着抖音小店走向了新的高潮&#xff0c;也带动着其它电商平台的发展&#xff0c;不断涌出新的平台…

计算机组成结构1

概念 计算机组成&#xff1a; 运算器、控制器、存储器、输出设备、输入设备 CPU&#xff1a;运算器控制器 运算器&#xff1a;算数逻辑单元ALU、累加寄存器AC、数据缓冲寄存器DR、状态条件寄存器PSW 控制器&#xff1a;指令寄存器IR、程序计数器PC、地址寄存器AR、指令译码器I…