从 C++11 中的容器元组中提取 value_type 的元组
Extracting a tuple of value_type from a tuple of containers in C++11
我有一个带有模板参数的函数,我知道它是几个不同元素类型的标准 C++ 容器的 std::tuple
。
如何从中提取 std::tuple
元素类型的类型?
例如,假设我有以下功能
template <typename TupOfCtrs>
void doStuff(const TupOfCtrs& tupOfCtrs) {
using TupOfElements = /*extract a tuple type by applying CtrT::value_type to each container in tupOfCtrs and combining the results into an std::tuple*/;
MyHelperClass<TupOfElements> helper;
}
我知道它是这样称呼的:
std::list<Foo> l {/*...*/};
std::vector<Bar> v {/*...*/};
std::deque<Baz> d {/*...*/};
auto tup = std::make_tuple(l, v, d);
在这种情况下,我希望将 TupOfElements
帮助程序类型定义为 std::tuple<Foo, Bar, Baz>
。
请注意,我实际上不需要 创建 元组,只需获取其类型即可。
如何实现,可能使用 Boost::Fusion 库?
您甚至可以在没有 Boost Fusion 的情况下以更简单的方式执行此操作,如下所示:
// Template which takes one type argument:
template <typename Tuple> struct TupOfValueTypes;
// Only provide a definition for this template for std::tuple arguments:
// (i.e. the domain of this template metafunction is any std::tuple)
template <typename ... Ts>
struct TupOfValueTypes<std::tuple<Ts...> > {
// This definition is only valid, if all types in the tuple have a
// value_type type member, i.e. the metafunction returns a type only
// if all types of the members in the std::tuple have a value_type
// type member, and a std::tuple can be constructed from these:
using type = std::tuple<typename Ts::value_type...>;
};
template <typename TupOfCtrs>
void doStuff(const TupOfCtrs& tupOfCtrs) {
using TupOfElements = typename TupOfValueTypes<TupOfCtrs>::type;
// ...
}
但是明确地为 std::tuple
指定 doStuff
当然更容易:
template <typename ... Ts>
void doStuff(const std::tuple<Ts...> & tupOfCtrs) {
using TupOfElements = std::tuple<typename Ts::value_type...>;
// ...
}
PS: 另请注意,在许多情况下,如果您只需要一个类型列表,则 std::tuple
class是一种矫枉过正,可能会稍微影响编译时间。就个人而言,我总是使用简单的 TypeList
结构:
template <typename ... Ts> struct TypeList
{ using type = TypeList<Ts...>; };
如果您希望 doStuff
接受 std::tuple
,请明确说明:
template <class... Ts>
void doStuff(std::tuple<Ts...> const& tupOfCtr) { ... }
一旦你有了那个参数包,只需取出 value_type
:
template <class... Ts>
void doStuff(std::tuple<Ts...> const& tupOfCtr)
{
using value_tuple = std::tuple<typename Ts::value_type...>;
// ...
}
我有一个带有模板参数的函数,我知道它是几个不同元素类型的标准 C++ 容器的 std::tuple
。
如何从中提取 std::tuple
元素类型的类型?
例如,假设我有以下功能
template <typename TupOfCtrs>
void doStuff(const TupOfCtrs& tupOfCtrs) {
using TupOfElements = /*extract a tuple type by applying CtrT::value_type to each container in tupOfCtrs and combining the results into an std::tuple*/;
MyHelperClass<TupOfElements> helper;
}
我知道它是这样称呼的:
std::list<Foo> l {/*...*/};
std::vector<Bar> v {/*...*/};
std::deque<Baz> d {/*...*/};
auto tup = std::make_tuple(l, v, d);
在这种情况下,我希望将 TupOfElements
帮助程序类型定义为 std::tuple<Foo, Bar, Baz>
。
请注意,我实际上不需要 创建 元组,只需获取其类型即可。
如何实现,可能使用 Boost::Fusion 库?
您甚至可以在没有 Boost Fusion 的情况下以更简单的方式执行此操作,如下所示:
// Template which takes one type argument:
template <typename Tuple> struct TupOfValueTypes;
// Only provide a definition for this template for std::tuple arguments:
// (i.e. the domain of this template metafunction is any std::tuple)
template <typename ... Ts>
struct TupOfValueTypes<std::tuple<Ts...> > {
// This definition is only valid, if all types in the tuple have a
// value_type type member, i.e. the metafunction returns a type only
// if all types of the members in the std::tuple have a value_type
// type member, and a std::tuple can be constructed from these:
using type = std::tuple<typename Ts::value_type...>;
};
template <typename TupOfCtrs>
void doStuff(const TupOfCtrs& tupOfCtrs) {
using TupOfElements = typename TupOfValueTypes<TupOfCtrs>::type;
// ...
}
但是明确地为 std::tuple
指定 doStuff
当然更容易:
template <typename ... Ts>
void doStuff(const std::tuple<Ts...> & tupOfCtrs) {
using TupOfElements = std::tuple<typename Ts::value_type...>;
// ...
}
PS: 另请注意,在许多情况下,如果您只需要一个类型列表,则 std::tuple
class是一种矫枉过正,可能会稍微影响编译时间。就个人而言,我总是使用简单的 TypeList
结构:
template <typename ... Ts> struct TypeList
{ using type = TypeList<Ts...>; };
如果您希望 doStuff
接受 std::tuple
,请明确说明:
template <class... Ts>
void doStuff(std::tuple<Ts...> const& tupOfCtr) { ... }
一旦你有了那个参数包,只需取出 value_type
:
template <class... Ts>
void doStuff(std::tuple<Ts...> const& tupOfCtr)
{
using value_tuple = std::tuple<typename Ts::value_type...>;
// ...
}