按名称创建 Protobuf 消息对象

Protobuf message object creation by name

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

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

C++ Protobuf 库已经为编译到二进制文件中的所有类型维护了一个 "descriptors" 池。

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();

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

我想对上述回复之一添加评论,但因为我没有足够的声誉计数,所以我将其添加为答案。同求见谅

我正在使用协议缓冲区 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'。