std :: forward_as_tuple将参数传递给2个构造函数

时间:2017-01-03 09:03:47

标签: c++ c++11 tuples

我想传递多个参数,以便在函数内构造两个对象,就像std::pair<T1, T2>(std::piecewise_construct, ...)一样。

所以我写了

template <typename Args0..., typename Args1...>
void f(std::tuple<Arg0> args0, std::tuple<Args1> args1) {
   Object0 alpha(...);
   Object1 beta(...);
   ...
}

所以我可以打电话

f(std::forward_as_tuple(..., ..., ...), std::forward_as_tuple(..., ...))

但我不知道如何构建Object0Object1。我检查了std::pair标准库的源代码,他们似乎使用复杂的内部函数来获取args0和args1的索引。你知道怎么做吗?

1 个答案:

答案 0 :(得分:3)

C ++ 17将有make_from_tuple来执行此操作,但您可以在C ++ 11中编写它。这是从cppreference窃取的C ++ 14版本(对于C ++ 11,您可以使用Implementation C++14 make_integer_sequencestd::index_sequence的实现。)

namespace detail {
template <class T, class Tuple, std::size_t... I>
constexpr T make_from_tuple_impl( Tuple&& t, std::index_sequence<I...> )
{
  return T(std::get<I>(std::forward<Tuple>(t))...);
}
} // namespace detail

template <class T, class Tuple>
constexpr T make_from_tuple( Tuple&& t )
{
    return detail::make_from_tuple_impl<T>(std::forward<Tuple>(t),
        std::make_index_sequence<std::tuple_size<std::decay_t<Tuple>>::value>{});
}

使用此实用程序,f的实现非常简单:

template <typename... Args0, typename... Args1>
void f(std::tuple<Args0...> args0, std::tuple<Args1...> args1) {
   auto alpha = make_from_tuple<Object0>(args0);
   auto beta = make_from_tuple<Object1>(args1);
}

为了使它更通用,我建议只推断那些元组的类型并完美转发它们:

template <typename T0, typename T1>
void f(T0&& args0, T1&& args1) {
   auto alpha = make_from_tuple<Object0>(std::forward<T0>(args0));
   auto beta = make_from_tuple<Object1>(std::forward<T1>(args1));
}

Live C++11 demo