我有很多protobuf消息,我目前使用手动编写的查找功能按名称生成消息。 由于随着项目的发展,消息越来越多,我已经厌倦了手工维护这些查找代码。
那么,有没有办法让这个过程自动化? 也许使用一个protoc插件,为protobuf代码添加一些代码,以便它可以自己注册?
答案 0 :(得分:10)
C ++ Protobuf库已经维护了一个"描述符池"对于编译成二进制文件的所有类型。
所以,你可以这样做:
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();
答案 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”。