How can I generate a tuple of N type T's?

后端 未结 3 1630
再見小時候
再見小時候 2020-12-19 14:28

I want to be able to write generate_tuple_type which would internally have a type alias type which would be std::tuple

相关标签:
3条回答
  • 2020-12-19 14:52

    Check the bottom of this link for an example:

    http://en.cppreference.com/w/cpp/utility/integer_sequence.

    You'll need to do a little more work to encapsulate the resulting tuple as a type alias, but the crucial construct here is std::integer_sequence and friends.

    0 讨论(0)
  • 2020-12-19 14:56

    You can use std::make_index_sequence to give you a pack long enough, and then just wrap it in the type you need. No recursion necessary:

    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;
    };
    
    0 讨论(0)
  • 2020-12-19 15:00

    Fairly straightforward recursive formulation:

    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;
    };
    

    Live example

    [Update]

    OK, so I was only thinking about modest values of N. The following formulation is more complex, but also significantly faster and less compiler-crushing for large arguments.

    #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" );
    }
    

    Live example

    This version performs at most 2*log(N)+1 template instantiations, assuming your compiler memoizes them. Proof left as an exercise for the reader.

    0 讨论(0)
提交回复
热议问题