Проверка правильности выражения вызова функции
std::result_of
вычисляет возвращаемый тип вызывающего выражения во время компиляции.
Как ссылка говорит, что если вызов плохо сформирован, то в результате создания std::result_of
возникает ошибка компиляции. Но предположим, что нам нужно проверить, правильно ли сформирован вызов, до получения типа результата.
Есть ли способ написать признак, который проверяет правильность формирования вызывающего выражения?
Что-то вроде:
template<typename F , typename... ARGS>
struct is_valid_call : public impl::is_valid_call<F,typelist<ARGS...>>
{};
namespace impl
{
struct sfinae_result{};
template<typename F , typename ARGS , typename ENABLED = sfinae_result>
struct is_valid_call;
template<typename F , typename... ARGS>
struct is_valid_call<F,typelist<ARGS...>,
decltype( std::declval<F>()(std::declval<ARGS>()...) )
> :
public std::true_type
{};
template<typename F , typename... ARGS>
struct is_valid_call<F,typelist<ARGS...>,sfinae_result> :
public std::false_type
{};
}
EDIT: Конечно, решение опубликовано не работает
Ответы
Ответ 1
Здесь что-то работает:
#include <type_traits>
#include <utility>
template<typename F, typename... Args>
struct is_valid_call {
private:
template<typename FF, typename... AA>
static constexpr auto check(int) ->
decltype( std::declval<FF>()(std::declval<AA>()...), std::true_type());
template<typename FF, typename... AA>
static constexpr std::false_type check(...);
public:
static constexpr bool value = decltype(check<F, Args...>(0)){};
};
#include <cstdio>
int main()
{
printf("%d", int (is_valid_call<decltype(&puts), const char*>::value));
printf("%d", int (is_valid_call<decltype(&puts), double>::value));
}
Выход: 10