如何生成 N 类型 T 的元组?
How can I generate a tuple of N type T's?
我希望能够编写内部具有类型别名type
的generate_tuple_type<int, 3>
,在这种情况下它将是std::tuple<int, int, int>
。
一些示例用法:
int main()
{
using gen_tuple_t = generate_tuple_type<int, 3>::type;
using hand_tuple_t = std::tuple<int, int, int>;
static_assert( std::is_same<gen_tuple_t, hand_tuple_t>::value, "different types" );
}
我怎样才能做到这一点?
相当简单的递归公式:
template<typename T, unsigned N, typename... REST>
struct generate_tuple_type
{
typedef typename generate_tuple_type<T, N-1, T, REST...>::type type;
};
template<typename T, typename... REST>
struct generate_tuple_type<T, 0, REST...>
{
typedef std::tuple<REST...> type;
};
实例
[更新]
好的,所以我只考虑N
的适度值。下面的公式更为复杂,但也明显更快,对大参数的编译器压缩更少。
#include <tuple>
template<typename /*LEFT_TUPLE*/, typename /*RIGHT_TUPLE*/>
struct join_tuples
{
};
template<typename... LEFT, typename... RIGHT>
struct join_tuples<std::tuple<LEFT...>, std::tuple<RIGHT...>>
{
typedef std::tuple<LEFT..., RIGHT...> type;
};
template<typename T, unsigned N>
struct generate_tuple_type
{
typedef typename generate_tuple_type<T, N/2>::type left;
typedef typename generate_tuple_type<T, N/2 + N%2>::type right;
typedef typename join_tuples<left, right>::type type;
};
template<typename T>
struct generate_tuple_type<T, 1>
{
typedef std::tuple<T> type;
};
template<typename T>
struct generate_tuple_type<T, 0>
{
typedef std::tuple<> type;
};
int main()
{
using gen_tuple_t = generate_tuple_type<int, 30000>::type;
static_assert( std::tuple_size<gen_tuple_t>::value == 30000, "wrong size" );
}
实例
这个版本最多执行2*log(N)+1个模板实例化,假设编译器将它们存储起来。作为练习留给读者的证据。
您可以使用std::make_index_sequence
给您一个足够长的包,然后将其包装成所需的类型。无需递归:
template <typename T, size_t N>
class generate_tuple_type {
template <typename = std::make_index_sequence<N>>
struct impl;
template <size_t... Is>
struct impl<std::index_sequence<Is...>> {
template <size_t >
using wrap = T;
using type = std::tuple<wrap<Is>...>;
};
public:
using type = typename impl<>::type;
};
这是一个带有Boost.Mp11的简单的一行程序。
#include <boost/mp11/algorithm.hpp>
int main()
{
using gen_tuple_t = boost::mp11::mp_repeat_c<std::tuple<int>, 3>;
using hand_tuple_t = std::tuple<int, int, int>;
static_assert( std::is_same_v<gen_tuple_t, hand_tuple_t>, "different types" );
}
查看此链接底部的示例:
http://en.cppreference.com/w/cpp/utility/integer_sequence.
您需要做更多的工作来将生成的元组封装为类型别名,但这里的关键构造是std::integer_sequence
和friends。
相关文章:
- 在C++中,如何通过几种类型从元组中选择多个元素
- 将元组类型扩展为可变参数模板?
- 时间复杂度 当具有复合数据类型(如元组或对)时?
- 创建派生自可变参数模板包的类型元组
- 枚举元组类型的最佳方法?
- 验证(使用 static_assert)元组类型是否遵循某种顺序(有状态编译时检查)
- C 元组类型用于功能参数类型
- C++ 如何根据部分特化制作 std::元组类型?
- 在运行时基于基础元组类型生成元素
- 模板函数,具有元组类型子集的相应参数
- 获取级联元组类型;组合result_of和tuple_cat
- 类元组类型(tuple,pair)的模板重载
- 如何将l值元组类型转换为值元组类型
- 使用C++元组类型作为映射中的键
- 转换元组类型
- 如何从旧的元组类型和boost中的类型创建新的元组类型
- 使用另一个构造一个具体的boost::元组类型
- 如何反转元组类型中元素类型的顺序
- 增强元组类型的hana大小
- 测试类型 V 是否属于没有变变量的元组类型<...>