合并多个地图
Merging multiple maps
我已经在 boost 邮件列表中询问过这个问题,但我似乎并不清楚自己的意图。也可能是我没有完全理解如何做到这一点。
我想在hana中合并多个地图,请看下面的代码示例:
constexpr auto m1 = hana::make_map(
hana::make_pair("key1"_s, hana::type_c<std::string>),
hana::make_pair("key2"_s, hana::type_c<std::string>)
);
constexpr auto m2 = hana::make_map(
hana::make_pair("key3"_s, hana::type_c<std::string>),
hana::make_pair("key4"_s, hana::type_c<std::string>),
hana::make_pair("key5"_s, hana::type_c<std::string>)
);
constexpr auto m3 = hana::make_map(
hana::make_pair("key6"_s, hana::type_c<std::string>),
hana::make_pair("key7"_s, hana::type_c<std::string>),
hana::make_pair("key8"_s, hana::type_c<std::string>)
);
我已经得到了如何为两张地图执行此操作的答案:
constexpr auto result = hana::fold_left(m1, m2, hana::insert);
constexpr auto expected = hana::make_map(
hana::make_pair("key1"_s, hana::type_c<std::string>),
hana::make_pair("key2"_s, hana::type_c<std::string>),
hana::make_pair("key3"_s, hana::type_c<std::string>),
hana::make_pair("key4"_s, hana::type_c<std::string>),
hana::make_pair("key5"_s, hana::type_c<std::string>)
);
当我查看文档时,我可以看到 fold_left 有 2 或 3 个参数。
看来我需要这样的东西:
fold_left(fold_left(m1, m3, hana::insert), m2, hana::insert);
template<typename Map...>
constexpr auto merge_multiple_maps(Map... args) {
// do something useful here...
}
但我不确定如何从这里开始,而且我在元编程方面仍然没有太多经验...
此致,马蒂斯
首先定义merge2
如下:
auto merge2 = [](auto&& m1, auto&& m2) {
return hana::fold_left(std::forward<decltype(m1)>(m1),
std::forward<decltype(m2)>(m2),
hana::insert);
};
然后,定义merge
作为merge2
的递归应用:
auto merge = [](auto&& m1, auto&& ...ms) {
return hana::fold_left(
hana::make_basic_tuple(std::forward<decltype(ms)>(ms)...),
std::forward<decltype(m1)>(m1),
merge2
);
};
我没有测试这个实现,但它应该给了你想法。如果不关心完美转发,可以把static_cast
全部drop掉;这只是为了提高运行时效率,以防您在地图中存储复制成本高但移动成本低的类型。此外,您将无法在 constexpr
上下文中使用它,因为 lambda 不能出现在常量表达式中。这将在 C++17 中修复,但现在您可以很容易地实现与这些 lambda 等效的函数对象。
[编辑:Hana 可能会在未来的某个时候实现此 merge
功能。]
[编辑:使用 std::forward
而不是 static_cast
。]
我已经在 boost 邮件列表中询问过这个问题,但我似乎并不清楚自己的意图。也可能是我没有完全理解如何做到这一点。
我想在hana中合并多个地图,请看下面的代码示例:
constexpr auto m1 = hana::make_map(
hana::make_pair("key1"_s, hana::type_c<std::string>),
hana::make_pair("key2"_s, hana::type_c<std::string>)
);
constexpr auto m2 = hana::make_map(
hana::make_pair("key3"_s, hana::type_c<std::string>),
hana::make_pair("key4"_s, hana::type_c<std::string>),
hana::make_pair("key5"_s, hana::type_c<std::string>)
);
constexpr auto m3 = hana::make_map(
hana::make_pair("key6"_s, hana::type_c<std::string>),
hana::make_pair("key7"_s, hana::type_c<std::string>),
hana::make_pair("key8"_s, hana::type_c<std::string>)
);
我已经得到了如何为两张地图执行此操作的答案:
constexpr auto result = hana::fold_left(m1, m2, hana::insert);
constexpr auto expected = hana::make_map(
hana::make_pair("key1"_s, hana::type_c<std::string>),
hana::make_pair("key2"_s, hana::type_c<std::string>),
hana::make_pair("key3"_s, hana::type_c<std::string>),
hana::make_pair("key4"_s, hana::type_c<std::string>),
hana::make_pair("key5"_s, hana::type_c<std::string>)
);
当我查看文档时,我可以看到 fold_left 有 2 或 3 个参数。
看来我需要这样的东西: fold_left(fold_left(m1, m3, hana::insert), m2, hana::insert);
template<typename Map...>
constexpr auto merge_multiple_maps(Map... args) {
// do something useful here...
}
但我不确定如何从这里开始,而且我在元编程方面仍然没有太多经验...
此致,马蒂斯
首先定义merge2
如下:
auto merge2 = [](auto&& m1, auto&& m2) {
return hana::fold_left(std::forward<decltype(m1)>(m1),
std::forward<decltype(m2)>(m2),
hana::insert);
};
然后,定义merge
作为merge2
的递归应用:
auto merge = [](auto&& m1, auto&& ...ms) {
return hana::fold_left(
hana::make_basic_tuple(std::forward<decltype(ms)>(ms)...),
std::forward<decltype(m1)>(m1),
merge2
);
};
我没有测试这个实现,但它应该给了你想法。如果不关心完美转发,可以把static_cast
全部drop掉;这只是为了提高运行时效率,以防您在地图中存储复制成本高但移动成本低的类型。此外,您将无法在 constexpr
上下文中使用它,因为 lambda 不能出现在常量表达式中。这将在 C++17 中修复,但现在您可以很容易地实现与这些 lambda 等效的函数对象。
[编辑:Hana 可能会在未来的某个时候实现此 merge
功能。]
[编辑:使用 std::forward
而不是 static_cast
。]