前言
本文对protobuf进行简单的介绍,适合新手入门了解练习使用。
一、protobuf语法详解
protobuf是google团队开发的用于高效存储和读取结构化数据的工具。什么是结构化数据呢,正如字面上表达的,就是带有一定结构的数据。比如电话簿上有很多记录数据,每条记录包含姓名、ID、邮件、电话等,这种结构重复出现。
xml、json也可以用来存储此类结构化数据,但是使用protobuf表示的数据能更加高效,并且将数据压缩得更小,大约是json格式的1/10,xml格式的1/20。
protobuf的语法详解可以通过这篇博客学习:https://blog.csdn.net/daaikuaichuan/article/details/105616844?utm_medium=distribute.pc_relevant.none-task-blog-baidujs_baidulandingword-1&spm=1001.2101.3001.4242
二、protobuf安装
先安装依赖项:
yum -y install gcc automake autoconf libtool make
在linux上安装,命令如下:
wget https://github.com/google/protobuf/releases/download/v3.6.1/protobuf-all-3.6.1.tar.gz
tar zxvf protobuf-all-3.6.1.tar.gz
./autogen.sh
./configure
make
make install
安装后更改环境变量:
vim /etc/profile
在文件的末尾添加如下的两行:
export PATH=$PATH:/usr/local/protobuf/bin/
export PKG_CONFIG_PATH=/usr/local/protobuf/lib/pkgconfig/
更改完成之后,执行如下命令立即执行:
source /etc/profile
配置动态链接库路径:
vim /etc/ld.so.conf
在文件中添加/usr/local/protobuf/lib(注意: 在新行处添加)
配置如下图所示:
更改完成之后,执行如下命令立即执行:
ldconfig
三、c++练习
先编写Person.proto文件如下所示:
syntax = "proto2";
package HUST_Person;
message Person{
required string name=1;
enum Gender{
BOY=0;
GIRL=1;
}
required Gender gender=2;
required int32 age=3;
required string ID=4;
message Home{
required string country=1;
required string province=2;
required string county=3;
required string addr=4;
}
optional Home home=5;
repeated string phone=6;
}
执行命令生成c++文件:
// $SRC_DIR: .proto 所在的源目录
// --cpp_out: 生成 c++ 代码
// $DST_DIR: 生成代码的目标目录
// xxx.proto: 要针对哪个 proto 文件生成接口代码
protoc -I=$SRC_DIR --cpp_out=$DST_DIR $SRC_DIR/xxx.proto
//实例:protoc --cpp_out=./ ./Person.proto
编写测试文件如下:
#include<iostream>
#include<string>
#include"Person.pb.h"
using namespace HUST_Person;
int main(){
Person person;
person.set_name("wyl");
person.set_gender(Person_Gender_BOY);
person.set_age(25);
person.set_id("M201870902");
Person_Home *home=new Person_Home();
home->set_country("china");
home->set_province("gansu");
home->set_county("minxian");
home->set_addr("**xiang**cun");
person.set_allocated_home(home);
person.add_phone("12256783456");
person.add_phone("11127222345");
std::cout<<"序列化后:"<<std::endl;
std::string str=person.SerializeAsString();
std::cout<<str<<std::endl;
std::cout<<"反序列化后:"<<std::endl;
Person copyPerson;
copyPerson.ParseFromString(str);
std::cout<<copyPerson.name()<<std::endl;
std::cout<<copyPerson.gender()<<std::endl;
std::cout<<copyPerson.age()<<std::endl;
std::cout<<copyPerson.id()<<std::endl;
std::cout<<copyPerson.home().country()<<std::endl;
std::cout<<copyPerson.home().province()<<std::endl;
std::cout<<copyPerson.home().county()<<std::endl;
std::cout<<copyPerson.home().addr()<<std::endl;
int size=copyPerson.phone_size();
for(int i=0;i<size;++i)
{
std::cout<<copyPerson.phone(i)<<std::endl;
}
return 0;
}
编译makefile如下:
libdir=/usr/lib/libprotobuf.so.17
cppfile=./Person.pb.cc ./Person_test.cpp
main:
g++ -g -o test.o -std=c++11 $(cppfile) $(libdir) -lpthread
clean:
rm -f test.o
最终效果如下: