Как std:: применять параметры forward без явного std::forward?

Рассмотреть возможность реализации std::apply:

namespace detail {
template <class F, class Tuple, std::size_t... I>
constexpr decltype(auto) apply_impl(F &&f, Tuple &&t, std::index_sequence<I...>) 
{
    return std::invoke(std::forward<F>(f), std::get<I>(std::forward<Tuple>(t))...);
}
}  // namespace detail

template <class F, class Tuple>
constexpr decltype(auto) apply(F &&f, Tuple &&t) 
{
    return detail::apply_impl(
        std::forward<F>(f), std::forward<Tuple>(t),
        std::make_index_sequence<std::tuple_size_v<std::decay_t<Tuple>>>{});
}

Почему при вызове функции (f) с набором параметров для передачи (t) нам не нужно выполнять std::forward на каждом элементе кортежа std::get<I>(std::forward<Tuple>(t))... в реализации?

2 ответа

Решение

Тебе не нужно std::forward каждый элемент, потому что std::get перегружен для rvalue-reference и lvalue-reference кортежа.

std::forward<Tuple>(t) даст вам либо lvalue (Tuple &) или значение (Tuple &&) и в зависимости от того, что вы получите, std::get даст вам T & (lvalue) или T && (Rvalue). Смотрите различные перегрузки std::get,


Немного подробностей о std::tuple а также std::get -

Как уже упоминалось в StoryTeller, каждый член кортежа является lvalue, независимо от того, был ли он создан из rvalue или lvalue, здесь не имеет значения:

double a{0.0};
auto t1 = std::make_tuple(int(), a);
auto t2 = std::make_tuple(int(), double());

Вопрос в том, является ли кортеж значимым? Если да, вы можете переместить его участника, если нет, вы должны сделать копию, но std::get уже позаботиться об этом, вернув члена с соответствующей категорией.

decltype(auto) a1 = std::get<0>(t1);
decltype(auto) a2 = std::get<0>(std::move(t1));

static_assert(std::is_same<decltype(a1), int&>{}, "");
static_assert(std::is_same<decltype(a2), int&&>{}, "");

Вернуться к конкретному примеру с std::forward:

template <typename Tuple>
void f(Tuple &&tuple) { // tuple is a forwarding reference
    decltype(auto) a = std::get<0>(std::forward<Tuple>(tuple));
}

f(std::make_tuple(int())); // Call f<std::tuple<int>>(std::tuple<int>&&);
std::tuple<int> t1;
f(t1); // Call f<std::tuple<int>&>(std::tuple<int>&);

В первом звонке f, тип a будет int&& так как tuple будет переслан как std::tuple<int>&& в то время как во втором случае его тип будет int& так как tuple будет переслан как std::tuple<int>&,

std::forward используется, чтобы убедиться, что все поступает на сайт вызова с правильной категорией значения.

Но каждый член кортежа является lvalue, даже если это кортеж rvalue Рекомендации.

Другие вопросы по тегам