按名称创建Protobuf消息对象

时间:2015-04-30 06:49:07

标签: c++ protocol-buffers

我有很多protobuf消息,我目前使用手动编写的查找功能按名称生成消息。 由于随着项目的发展,消息越来越多,我已经厌倦了手工维护这些查找代码。

那么,有没有办法让这个过程自动化? 也许使用一个protoc插件,为protobuf代码添加一些代码,以便它可以自己注册?

3 个答案:

答案 0 :(得分:10)

C ++ Protobuf库已经维护了一个"描述符池"对于编译成二进制文件的所有类型。

https://developers.google.com/protocol-buffers/docs/reference/cpp/google.protobuf.descriptor#DescriptorPool.generated_pool.details

所以,你可以这样做:

google::protobuf::Descriptor* desc =
    google::protobuf::DescriptorPool::generated_pool()
        ->FindMessageTypeByName("mypkg.MyType");
assert(desc != NULL);

该库还维护了一个对象,该对象可用于构造任何编译类型的实例,给定其描述符:

https://developers.google.com/protocol-buffers/docs/reference/cpp/google.protobuf.message#MessageFactory.generated_factory.details

所以你要这样做:

google::protobuf::Message* message =
    google::protobuf::MessageFactory::generated_factory()
        ->GetPrototype(desc)->New();

答案 1 :(得分:2)

不太可能这种方法不能用作创建任何消息实例的通用方法。只有在此类型的消息至少实例化一次后(例如,在generated_pool()时),MyMessageType* msg = new MyMessageType()中才会显示消息类型描述,因此FindMessageTypeByName永远不会找到消息类型尚未实例化的消息。

答案 2 :(得分:0)

我想对以上答复之一添加评论,但是由于我没有足够的声誉,因此我将其添加为答案。寻求原谅。

我正在使用协议缓冲区3.6.1,我注意到生成的.pb.cc文件中的一些代码与Kenton可能指向的内容有关。

namespace protobuf_foo_5fcp_5fplayer_5fcommon_5fevent_5ftypes_2eproto {
void InitDefaults() {
}
//...
//...

// Force AddDescriptors() to be called at dynamic initialization time.
struct StaticDescriptorInitializer {
  StaticDescriptorInitializer() {
    AddDescriptors();
  }
} static_descriptor_initializer;
}  // namespace protobuf_foo_5fcp_5fplayer_5fcommon_5fevent_5ftypes_2eproto

似乎从未调用全局变量static_descriptor_initializer。 我通过如下修改代码并验证从未调用调用cout的消息来发现这一点!

//...
//...

#include <iostream>

// Force AddDescriptors() to be called at dynamic initialization time.
struct StaticDescriptorInitializer {
  StaticDescriptorInitializer() {
    AddDescriptors();
    std::cout << "##################> DESCRIPTORS ADDED\n";
  }
} static_descriptor_initializer;

现在我想我必须找出g ++(我正在使用)中是否有一个选项来导致在应用程序启动序列期间构造“ static_descriptor_initializer”。