Qt/C++ 序列化(protobuf方式)

一、序列化介绍

序列化是将需要的内存对象转换为字节流或者其他数据格式的过程,以便保存到文件或者传输。反序列化则是将数据还原到内存对象。

序列化后存档格式有二进制数据、XML或JSON文件等。可以按照自己的需求进行序列化。

二、protobuf介绍

Protocal Buffers(简称protobuf)是谷歌使用的一项技术标准,可以将数据保存为二进制格式,完成序列化和反序列化。

优势与缺点

(优势)跨语言、跨平台、可扩展,比XML、JSON等占用空间小,效率高。

(缺点)可读性比XML,JSON等效果差。

JSON、XML是很多行业标准,而protobuf只是Google公司的内部标准。

三、工具包生成

1. 工具包下载

在github上下载,地址:https://github.com/protocolbuffers/protobuf/releases

选择一个C++版本,下载后解压。

2. 库生成

下载cmake工具,将cmake工具configure配置好后,执行后生成protobuf相关的解决方案

打开protobuf.sln,通过你的VS编译工具,编译后生成工具及库。

四、序列化流程

1. proto文件

新建一个文本文件,为了将数据实现序列化,需要将数据整理成message(相当于结构体)的基本类型写入文件。数据保存后扩展名改为proto文件,proto文件示例如下:

上图中

(1)syntax = "proto3" 指protobuf的版本号,当前主版本为3。

(2)可以声明一个package XX,后续利用它生成的源文件会将XX替换为命名空间,将序列化的模块加入到该命名空间中。

(3)定义message内容,message内会有多个变量,每个变量有个对应的编号,而且不同变量的编号不能重复。
protobuf中常用的数据类型:

数据类型 说明
bool 布尔类型
double 64位浮点数
float 32位浮点数
int32 32位整数
int64 64位整数
uint64 64位无符号整数
sint32 32位整数,处理负数效率更高
sint64 64位整数,处理负数效率更高
string 只能处理ASCII字符
bytes 用于处理多字节的语言字符
enum 枚举类型

2. 生成序列化接口文件

打开cmd控制台,切换到前面生成protoc.exe路径,输入命令protoc ./test.proto --cpp_out=./,将会从XX.proto 文件生成相应语言(cpp,c#, java)的接口文件,然后加入到我们的项目中进行使用。

生成的文件名为Test.pb.h,Test.pb.cc

3. Qt中的配置使用

(1)定义PROTOBUF_USE_DLLS使用

(2)从原始protobuf工具包中的头文件文件夹(src)拷贝到当前项目中,并添加

(3)将你编译环境生成的libprotobuf库及路径包含引用

bash 复制代码
#switch protobuf
DEFINES += PROTOBUF_USE_DLLS

SOURCES += $$PWD/proto/test.pb.cc \
    $$PWD/pserializer.cpp

HEADERS += $$PWD/proto/test.pb.h \
    $$PWD/pserializer.h

INCLUDEPATH += $$PWD/src

win32:CONFIG(release, debug|release): LIBS += -L$$PWD/lib/release -llibprotobuf
else:win32:CONFIG(debug, debug|release): LIBS += -L$$PWD/lib/debug -llibprotobufd

4. 序列化和反序列化代码

serializeToString函数:序列化为字符串

ParseFromString函数:从字符串中反序列化

cpp 复制代码
void PSerializer::serializer()
{
    // 序列化
    testPackage::Person person;
    person.set_id(0);
    person.set_name("lihua");
    person.set_sex("man");
    person.set_age(25);

    std::string string;
    if (!person.SerializeToString(&string)) {
        qDebug() << " Serialize Failed";
    }
    qDebug() << "string" << string.c_str();

    //反序列化
    testPackage::Person person1;
    if(!person1.ParseFromString(string))
    {
        qDebug() << " Parse Failed";
    }
    qDebug() << "person1.id"   << person1.id();
    qDebug() << "person1.name" << person1.name().c_str();
    qDebug() << "person1.sex"  << person1.sex().c_str();
    qDebug() << "person1.age"  << person1.age();
}

至此,我们完成一个简单的protobuf序列化和反序列化的实现。

相关推荐
捕鲸叉3 小时前
创建线程时传递参数给线程
开发语言·c++·算法
A charmer3 小时前
【C++】vector 类深度解析:探索动态数组的奥秘
开发语言·c++·算法
Peter_chq3 小时前
【操作系统】基于环形队列的生产消费模型
linux·c语言·开发语言·c++·后端
青花瓷4 小时前
C++__XCode工程中Debug版本库向Release版本库的切换
c++·xcode
幺零九零零6 小时前
【C++】socket套接字编程
linux·服务器·网络·c++
捕鲸叉6 小时前
MVC(Model-View-Controller)模式概述
开发语言·c++·设计模式
徒步僧7 小时前
ThingsBoard规则链节点:RPC Call Reply节点详解
qt·microsoft·rpc
Dola_Pan7 小时前
C++算法和竞赛:哈希算法、动态规划DP算法、贪心算法、博弈算法
c++·算法·哈希算法
yanlou2337 小时前
KMP算法,next数组详解(c++)
开发语言·c++·kmp算法
小林熬夜学编程7 小时前
【Linux系统编程】第四十一弹---线程深度解析:从地址空间到多线程实践
linux·c语言·开发语言·c++·算法