我在處理任意類型的反轉模板參數時遇到了這個問題。
喬納森Wakely的答案偉大的元組,但在其他人需要反轉任何類型,即T<P1, P2, ..., Pn>
至T<Pn, Pn-1, ..., P1>
,這是我想出了(Reversal logic taken from here)。
namespace Details
{
/// Get the base case template type `T<>` of a templated type `T<...>`
template<typename>
struct templated_base_case;
template <template<typename...> class T, typename... TArgs>
struct templated_base_case<T<TArgs...>>
{
using type = T<>;
};
/// Inner reverse logic.
///
/// Reverses the template parameters of a templated type `T` such
/// that `T<A, B, C>` becomes `T<C, B, A>`.
///
/// Note that this requires `T<>` to exist.
template<
typename T,
typename = typename templated_base_case<T>::type>
struct reverse_impl;
template<
template <typename...> class T,
typename... TArgs>
struct reverse_impl<
typename templated_base_case<T<TArgs...>>::type,
T<TArgs...>>
{
using type = T<TArgs...>;
};
template<
template<typename...> class T,
typename first,
typename... rest,
typename... done>
struct reverse_impl<
T<first, rest...>,
T<done...>>
{
using type = typename reverse_impl <T<rest...>, T<first, done...>>::type;
};
/// Swap template parameters of two templated types.
///
/// `L<A, B, C> and R<X, Y, Z>` become `L<X, Y, Z> and R<A, B, C>`.
template<typename L, typename R>
struct swap_template_parameters;
template<
template<typename...> class L,
template<typename...> class R,
typename... x,
typename... y>
struct swap_template_parameters<L<x...>, R<y...>>
{
using left_type = L<y...>;
using right_type = R<x...>;
};
}
/// Parameter pack list of types
template <typename... Args>
struct type_list { };
/// Reverses the arguments of a templates type `T`.
///
/// This uses a `type_list` to allow reversing types like std::pair
/// where `std::pair<>` and `std::pair<T>` are not valid.
template<typename T>
struct reverse_type;
template<template<typename...> class T, typename... TArgs>
struct reverse_type<T<TArgs...>>
{
using type = typename Details::swap_template_parameters<
T<TArgs...>,
typename Details::reverse_impl<type_list<TArgs...>>::type>::left_type;
};
某些實現邏輯可以合併,但我試圖在這裏儘可能清楚。
reverse_type
可以應用到元組:
using my_tuple = std::tuple<int, bool, char>;
static_assert(
std::is_same<
typename reverse_type<my_typle>::type,
std::tuple<char, bool, int>>::value,
"");
或其他類型的:
/// Standard collections cannot be directly reversed easily
/// because they take default template parameters such as Allocator.
template<typename K, typename V>
struct simple_map : std::unordered_map<K, V> { };
static_assert(
std::is_same<
typename reverse_type<simple_map<std::string, int>>::type,
simple_map<int, std::string>>::value,
"");
Slightly more detailed explanation。
我不認爲你需要遞歸做到這一點,[tuple_cat(http://www.cplusplus.com/reference/tuple/tuple_cat/),但你爲什麼要扭轉一個元組 – aaronman