gpt4 book ai didi

c++ - 在 C++ 中使用 MessagePack 反序列化异构映射

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:48:10 28 4
gpt4 key购买 nike

我在 C++ 中使用 MessagePack,我正在尝试反序列化这个 Python 映射的等效项:

{'metadata': {'date': '2014-06-25', 'user_id': 501},
'values': [3.0, 4.0, 5.0],
'version': 1}

顶级对象是一个带有字符串键的映射,但值是完全不同的类型。我的代码提前知道对象的结构应该是什么;我应该能够声明一个整数,然后告诉我的反序列化代码,“version 键的值是一个整数,因此将该整数的值放入该内存地址。”

问题是我什至不确定如何达到我的 C++ 代码可以将此结构视为映射的地步。我希望做类似的事情

msgpack::unpacker unpacker;
// ...copy the data into unpacker's buffer...

msgpack::unpacked message;
std::map<std::string, anything> output_map;

unpacker.next(&message);
msgpack::object obj = message.get();
obj.convert(&output_map);

int version_number = output_map.at("version");

是否有任何可能的类型(anything)可以在这里工作? MessagePack 文档只有一些简单的示例,并且 this blog post更好,但不涵盖此用例。

最佳答案

您可以使用 boost::variant 来做到这一点。要实现递归结构,可以使用 oost::make_recursive_variant 如下:

typedef boost::make_recursive_variant<
std::string,
std::map<boost::recursive_variant_, boost::recursive_variant_>,
std::vector<boost::recursive_variant_>,
int,
double
>::type variant_t;

这是一个文档: http://www.boost.org/doc/libs/1_55_0/doc/html/variant/tutorial.html#variant.tutorial.recursive.recursive-variant

您还需要编写一个转换器,将 msgpack::object 转换为 variant_t,反之亦然,如下所示:

// Custom converter for variant_t
namespace msgpack {

// Convert from msgpacl::object to variant_t.
inline variant_t& operator>>(object const& o, variant_t& v) {
switch(o.type) {
case type::MAP:
v = std::map<variant_t, variant_t>();
o.convert(boost::get<std::map<variant_t, variant_t> >(&v));
break;
case type::ARRAY:
v = std::vector<variant_t>();
o.convert(boost::get<std::vector<variant_t> >(&v));
break;
case type::POSITIVE_INTEGER:
v = int();
o.convert(boost::get<int>(&v));
break;
case type::DOUBLE:
v = double();
o.convert(boost::get<double>(&v));
break;
case type::RAW:
v = std::string();
o.convert(boost::get<std::string>(&v));
break;
default:
break;
}
return v;
}


// Convert from variant_t to msgpacl::object.
template <typename Stream>
struct packer_imp:boost::static_visitor<void> {
template <typename T>
void operator()(T const& value) const {
o_.pack(value);
}
packer_imp(packer<Stream>& o):o_(o) {}
packer<Stream>& o_;
};

template <typename Stream>
inline packer<Stream>& operator<< (packer<Stream>& o, const variant_t& v)
{
boost::apply_visitor(packer_imp<Stream>(o), v);
return o;
}

} // namespace msgpack

您可以从 gist 获得完整的示例代码: https://gist.github.com/redboltz/672c5af16b2907488977我在示例中使用了 C++11 功能,因此您需要添加 -std=c++11 选项。

关于c++ - 在 C++ 中使用 MessagePack 反序列化异构映射,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24412133/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com