c++ protobuf 可能会遇到的坑

1.发现存在内存泄露。

程序退出时记得调用:

google::protobuf::ShutdownProtobufLibrary();

这里一定是在程序退出时调用,如果调用后又使用了 protobuf 会出现异常,因为protobuf 中使用构造 会有创建指针,会存入 ShutdownData;

ShutdownProtobufLibrary 源码:

void ShutdownProtobufLibrary() {
  // This function should be called only once, but accepts multiple calls.
  static bool is_shutdown = false;
  if (!is_shutdown) {
    delete internal::ShutdownData::get();
    is_shutdown = true;
  }
}

解决方案:

定义成全局,全局只释放一次

struct A {
    ~A(){
         google::protobuf::ShutdownProtobufLibrary() ;
    }
};
A  _globalProtobuf;

2.内存有异常:

  可能是:protobuf 中的嵌套消息的使用临时变量例:

string sn="1111";

string Algo="3333";

request.set_sn(sn);
request.set_algo(Algo);

如果在其它地方使用可能会有异常;需要去new,退去时记得 release_eventcode

 PointProtos::Event *event = mUploadLogInfoData->add_events();
 string* ans = event->mutable_eventcode();
 (*ans)=key;
 string* tvalue = event->mutable_eventcode();//
 (*tvalue)= value ;

  如果是对象一样:
for(int index = 0;index<info.answer_size();index++)
{
    Detail * detail = rsp.add_detail();
    Answer* ans = detail->mutable_answer();
    Answer temp_ans = info.answer(index);
    ans->copyFrom(temp_ans);
}
 

总结:protobuf 中的嵌套消息的使用主要对set_allocated_和mutable_的使用

1 使用set_allocated_,赋值的对象需要new出来,不能用局部的,这里保存的是对象的指针。

2 使用mutable_,赋值时候,可以使用局部变量,因为在调用的时,内部做了new操作。
 

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
C++ protobufProtocol Buffers)是Google开发的一种数据序列化协议,可以将数据结构序列化为二进制格式,以便于在不同平台之间进行数据传输和存储。C++ protobuf提供了一种方便、高效和可扩展的数据序列化方式,可以在不同的系统和编程语言之间进行数据交换。 使用C++ protobuf需要进行以下步骤: 1. 定义消息类型 首先,需要定义消息类型,也就是要序列化和反序列化的数据结构。可以使用.proto文件来定义消息类型,然后使用protobuf编译器生成对应的C++代码。例如,下面是一个简单的.proto文件: ``` syntax = "proto3"; message Person { string name = 1; int32 age = 2; string email = 3; } ``` 这个.proto文件定义了一个Person消息类型,包含三个字段:name、age和email。 2. 生成C++代码 使用protobuf编译器生成对应的C++代码,可以使用以下命令: ``` protoc --cpp_out=. person.proto ``` 这个命令在当前目录下生成person.pb.h和person.pb.cc两个文件,它们包含了自动生成的C++代码,用于对Person消息类型进行序列化和反序列化。 3. 序列化 要将消息序列化为二进制格式,需要创建一个protobuf消息对象,并将数据赋值给它的字段。例如,下面是一个简单的例子: ``` #include "person.pb.h" #include <iostream> int main() { Person person; person.set_name("Alice"); person.set_age(30); person.set_email("alice@example.com"); std::string serialized; person.SerializeToString(&serialized); std::cout << "Serialized: " << serialized << std::endl; return 0; } ``` 在这里,我们创建了一个Person对象,并对它的字段进行了赋值。然后,我们使用SerializeToString函数将Person对象序列化为二进制格式,并将结果存储在serialized字符串中。 4. 反序列化 要从二进制格式中反序列化消息,需要使用ParseFromString函数,将二进制数据解析为protobuf消息对象。例如,下面是一个简单的例子: ``` #include "person.pb.h" #include <iostream> int main() { std::string serialized = ...; // 从某个地方获取二进制数据 Person person; person.ParseFromString(serialized); std::cout << "Name: " << person.name() << std::endl; std::cout << "Age: " << person.age() << std::endl; std::cout << "Email: " << person.email() << std::endl; return 0; } ``` 在这里,我们将二进制数据存储在serialized字符串中,并使用ParseFromString函数将其解析为Person对象。然后,我们可以使用person对象的getter函数获取各个字段的值。 这就是使用C++ protobuf进行数据序列化和反序列化的基本步骤。当然,protobuf还提供了很多其他的功能,例如使用动态消息类型、使用流式API等等。具体使用方式可以参考protobuf官方文档。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

恋恋西风

up up up

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值