Как мы можем проверить, может ли выражение определенного типа вызываться с помощью prvalue?

С C++17 у нас есть новое is_invocable и представьте новые ценности, которые на самом деле не являются ценностями.

Это позволяет вам создавать объект без необходимости сначала логически его конструировать, а затем исключать конструкцию.

Я столкнулся с проблемой, когда с помощью std::is_invocable чтобы проверить, можете ли вы что-то вызывать, и правила prvalue, кажется, сталкиваются:

struct no_move {
  no_move(no_move&&)=delete;
  explicit no_move(int) {}
};
void f( no_move ) {}

Теперь мы можем спросить, если f может быть вызван с использованием prvalue типа no_move?

f( no_move(1) )

std::is_invocable< decltype(&f), no_move > не работает, потому что использует std::declval<no_move>() который похож на xvalue no_move&& не тип значения no_move,

В C++14 это было то же самое, но гарантированное elision делает некоторые функции вызываемыми с помощью xvalue (т. Е. " T&& ") и другие с типами значений T,

Есть ли альтернатива, или мы должны изобрести собственную черту, чтобы справиться с этим делом?

(В теоретическом мире, где std::declval<T> возвращенный T вместо T&&, is_invocable я бы, наверное, поступил правильно).

2 ответа

Решение

Есть ли альтернатива, или мы должны изобрести собственную черту, чтобы справиться с этим делом?

Да, вам просто нужно написать свою собственную черту, которая не использует declval, Если у вас есть std::is_detected валяется (что я знаю, вы, конечно, делаете)

template <typename T> T make();

template <typename F, typename... Args>
using invoke_result_t = decltype(std::declval<F>()(make<Args>()...));
//                               ^^^^^^^^^^^^^     ^^^^^

template <typename F, typename... Args>
using is_invocable = std::is_detected<invoke_result_t, F, Args...>;

Сюда, std::is_invocable<decltype(f), no_move> является false_type, но is_invocable<decltype(f), no_move)> является true_type,

Я намеренно использую declval<F>() для функции вместо make чтобы разрешить использование decltype(f) Вот. В самом деле, invoke_result_t должно быть более сложным и "делать правильные вещи" для указателей на членов и т. д. Но это по крайней мере простое приближение, которое указывает на жизнеспособность этого подхода.

Вы неправильно используете концепцию Invocable. Эта концепция означает не более, чем способность использовать std::invoke на данную функцию и предоставленные аргументы.

Ты не можешь сделать std::invoke(f, no_move(1)), так как это вызовет копирование / перемещение перенаправленного аргумента. Невозможно использовать значение prvalue в качестве параметра при переадресации вызова, например invoke, Вы можете передать значение для переадресации вызова, но возможный вызов данной функции получит значение x.

Это хорошая причина избегать использования неподвижных типов в качестве параметров-значений в функциях. Возьми их const& вместо.

C++ не имеет черты типа, чтобы увидеть, можно ли вызывать функцию с определенными параметрами желаемым способом.

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