按名称创建 Protobuf 消息对象
Protobuf message object creation by name
我有很多 protobuf 消息,我目前使用手动编写的查找函数按其名称生成消息。
随着项目的发展,消息越来越多,我已经厌倦了手动维护这个查找代码。
那么,有没有办法使这个过程自动化?
也许使用 protoc 插件向 protobuf 代码添加一些代码,以便它可以自行注册?
C++ Protobuf 库已经为编译到二进制文件中的所有类型维护了一个 "descriptors" 池。
所以,你可以这样做:
google::protobuf::Descriptor* desc =
google::protobuf::DescriptorPool::generated_pool()
->FindMessageTypeByName("mypkg.MyType");
assert(desc != NULL);
该库还维护一个对象,可用于构造任何编译类型的实例,给定其描述符:
所以你会这样做:
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'。
我有很多 protobuf 消息,我目前使用手动编写的查找函数按其名称生成消息。 随着项目的发展,消息越来越多,我已经厌倦了手动维护这个查找代码。
那么,有没有办法使这个过程自动化? 也许使用 protoc 插件向 protobuf 代码添加一些代码,以便它可以自行注册?
C++ Protobuf 库已经为编译到二进制文件中的所有类型维护了一个 "descriptors" 池。
所以,你可以这样做:
google::protobuf::Descriptor* desc =
google::protobuf::DescriptorPool::generated_pool()
->FindMessageTypeByName("mypkg.MyType");
assert(desc != NULL);
该库还维护一个对象,可用于构造任何编译类型的实例,给定其描述符:
所以你会这样做:
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'。