- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
在protobuf中,我们有几个选项来实现继承。 “嵌套扩展”就是其中之一: http://www.indelible.org/ink/protobuf-polymorphism/
这里有趣的是如何读取序列化文件。我们必须创建一个 Map 以将 Animal.type 与其扩展标识符相对应,以便将动物转换为正确的 Dog 或 Cat。但是,在上面网站提供的示例中,使用的语言是Python。这意味着可以在不指定键类型或值类型的情况下初始化 Map。而且效果很好:
# Unpack the serialized bytes.
animal = Animal()
animal.ParseFromString(bytes)
# Determine the appropriate extension type to use.
extension_map = { Animal.Cat: Cat.animal, Animal.Dog: Dog.animal }
extension = animal.Extensions[extension_map[animal.type]]
但是,为了在C++中实现这样的映射,键类型和值类型是强制性的。那么,我应该使用什么类型的值才能将两个不同的扩展标识符存储到同一个映射中?
Map<Animal::Type, ::google::protobuf::internal::ExtensionIdentifier>
?
不幸的是,这显然行不通。
我还要在这里复制粘贴写作范式: 从 animals_pb2 导入 *
# Construct the polymorphic base message type.
animal = Animal()
animal.type = Animal.Cat
# Create the subclass type by referencing the appropriate extension type.
# Note that this uses the self-referential field (Cat.animal) from within
# nested message extension.
cat = animal.Extensions[Cat.animal]
cat.declawed = True
# Serialize the complete message contents to a string. It will end up
# looking roughly like this: [ type [ declawed ] ]
bytes = animal.SerializeToString()
Extensions() 函数可以让我们使用扩展的标识符来获取它的扩展。
最佳答案
如果我对问题的理解正确,您希望将构造的扩展对象的映射存储在映射中,以便在解析消息后访问。
在这种情况下,有 variant和 any在 boost 中键入,poco libraries还有一些。您可以使键类型固定(即枚举类型或字符串),并将值类型设为变体类型:
#include <boost/any.hpp>
#include <map>
#include <iostream>
#include <string>
#include <memory>
struct Animal {
std::string type;
};
struct Dog : public Animal {
constexpr static const char* TYPE = "Dog";
void bark() const {
std::cout<<"bow"<<std::endl;
}
};
struct Cat : public Animal {
constexpr static const char* TYPE = "Cat";
void meow() const {
std::cout<<"meow"<<std::endl;
}
};
从消息中获取扩展的工厂:
template <typename Animal,typename Message>
std::unique_ptr<Animal> get_extension(Message m) {
try {
Animal a( boost::any_cast<Animal>(m[Animal::TYPE]) );
//for simplicity
return std::unique_ptr<Animal>(new Animal(a));
} catch (...) {
return std::unique_ptr<Animal>();
}
}
和用法:
int main()
{
// simulation of a protobuf message
std::map<std::string,boost::any> m;
m[Dog::TYPE] = Dog();
m[Cat::TYPE] = Cat();
// the generic interface to the message extensions
auto dog = get_extension<Dog>(m);
if (dog)
dog->bark();
auto cat = get_extension<Cat>(m);
if (cat)
cat->meow();
}
更新:version 2具有动物通用界面
实际任务可能是您有一 strip 有扩展名的消息,并且您想要动态创建对象。在第二个版本中,您还可以通过相同的界面“与动物交谈”:
struct Animal {
virtual void speak() const = 0;
virtual ~Animal(){}
};
struct Dog : public Animal {
constexpr static const char* TYPE = "Dog";
virtual void speak() const {
std::cout<<"bow"<<std::endl;
}
};
// ... etc
类型反序列化动物的简单工厂:
struct AnimalRegistry {
std::map<std::string , std::function<std::unique_ptr<Animal>()>> creators;
template<typename A>
void register_creator() {
creators[A::TYPE] = []() { return std::unique_ptr<Animal>(new A); };
}
template<typename Message>
std::unique_ptr<Animal> create(Message m) {
return creators[m.animal_type]();
}
};
用法略有不同:
int main()
{
AnimalRegistry registry;
registry.register_creator<Dog>();
registry.register_creator<Cat>();
Message received_message { "Dog" /*this is a part of your protobuf*/ };
auto dog = registry.create(received_message);
if (dog)
dog->speak();
Message another_message { "Cat" };
auto cat = registry.create(another_message);
if (cat)
cat->speak();
}
关于c++ - 在 C++ 中,如何创建值为 protobuf 扩展标识符的映射,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19975595/
C++ Primer 说: The identifier we define in our programs may not contain 2 consecutive underscores, no
标识符术语在文档 alongside constants 中定义。 , 使用几乎相同的用例,尽管术语在运行时计算它们的值,而常量在编译时得到它。潜在地,这可能会使术语使用全局变量,但这是一个遥远而丑陋
我想知道,.Net 标识符中接受哪些字符? 不是 C# 或 VB.Net,而是 CLR。 我问这个的原因是我正在查看 yield return 语句是如何实现的 (C# In Depth),并看到
在PowerShell中,当我专门使用Active Directory时,通常会编译一个包含一组人群列表的对象,通常使用$x = get-adgroup -filter {name -like "*"
使用 hibernate 时: 我必须为每个实体指定一个 ID 或复合 ID,如果我想使用没有任何主键且没有复合键的表怎么办... 提前致谢 最佳答案 没有键的表不是一个好的关系模型。我不会推荐它。
所以我有一些代码正在尝试编译,但我不断收到此错误: 3SATSolver.java:3: expected 这是代码。我只是没有看到什么吗? import java.util.ArrayList;
我正在寻找有关 C 标准(C99 和/或 C11)部分内容的一些说明,主要是关于标识符的使用。 上下文是一个完整的C99标准库的实现,我希望它完全符合标准。 基本问题是:C 标准允许我在多大程度上声明
我有这个 Shader.h 文件,我正在用这段代码制作它: #include #include #include #include in vec2 TexCoords; out vec4 co
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
这是我的代码: #include "stdafx.h" #include #include #include #include using namespace std; int _tmain(
pthread_create() 的第一个参数是一个thread 对象,用于标识新创建的线程。但是,我不确定我是否完全理解其中的含义。 例如,我正在编写一个简单的聊天服务器并且我计划使用线程。线程会随
我想从我的标识符中获得匹配项。 我在 {/describe:foo} 中有一个这样的字符串,我正在尝试匹配 {/describe:} 以返回 foo,但我没有得到正确的正则表达式,会有人介意指出我做错
我遇到了一个奇怪的问题,我似乎找不到答案,但我想我不妨问问。 我有一个执行碰撞检查的抽象类,它有一个“更新”函数,以及“updateX”和“updateY”函数。 class MapCollidabl
我正在尝试创建一个程序来将所有文件从一个目录复制到另一个目录。但我遇到了一个基本问题。它说当我尝试在第 52 行编译时需要标识符。 public bool RecursiveCopy() {
1>cb.c(51): error C2061: syntax error : identifier 'SaveConfiguration' 1>cb.c(51): error C2059: synt
我刚刚发现命名变量 arguments 是个坏主意。 var arguments = 5; (function () { console.log(arguments); })(); Outpu
我们对我们的网站进行了安全测试,并发现了一个漏洞。 问题 If the session identifier were known by an attacker who had access to t
为了估计程序在一次内核启动中可以处理多少数据,我尝试使用 cudaMemGetInfo() 获取一些内存信息。但是,编译器告诉我: 错误:标识符“cudaMemGetInfo”未定义 cudaGetD
我发现我需要使用 xpath 查询来定位几乎是 regexp 类型的字符串,但无法看到如何管理它。我正在使用的当前查询是: $result = $xpath->query('//ul/li[sta
我正在创建我的学生计划表的虚拟版本,它基本上可以让你记下你有哪些科目的作业。 这是界面: 用户从组合框中选择主题,并在相邻的备忘录中输入一些注释。完成后,他们将单击“保存”按钮,将其保存到 .ini
我是一名优秀的程序员,十分优秀!