检查函数是否可调用
Check if a function is callable
我使用的is_callable
结构定义如下
template <typename F, typename... Args>
struct is_callable {
template <typename U>
static auto test(U* p) -> decltype((*p)(std::declval<Args>()...), void(), std::true_type());
template <typename U>
static auto test(...) -> decltype(std::false_type());
static constexpr bool value = decltype(test<F>(nullptr))::value;
};
我正在使用它来测试声明为的 lambda:
template <typename T>
struct runner {
T t;
template <typename F, typename = typename std::enable_if<is_callable<F, T&>::value || is_callable<F, T&&>::value>::type>
void run(F&& f) {
return f(t);
}
};
runner<int> a{0};
a.run([&] (auto& x) {
x++;
});
为什么在 AppleClang 的 enable_if
上编译失败? auto
不应该正确推断吗?
您的 true_type
test
案例看起来不对。无论如何,您的代码比需要的更复杂。尝试以下最小工作示例:
#include <utility>
template<typename F, typename...Args> struct is_callable {
template<typename F2, typename...Args2> static constexpr std::true_type
test(decltype(std::declval<F2>()(std::declval<Args2>()...)) *) { return {}; }
template<typename F2, typename...Args2> static constexpr std::false_type
test(...) { return {}; }
static constexpr bool value = decltype(test<F, Args...>(nullptr))::value;
};
void f0();
static_assert(is_callable<decltype(f0)>::value, "f0()");
static_assert(!is_callable<decltype(f0), int>::value, "f0(0)");
int f1(int);
static_assert(!is_callable<decltype(f1)>::value, "f1()");
static_assert(is_callable<decltype(f1), int>::value, "f1(0)");
static_assert(!is_callable<decltype(f1), int, int>::value, "f1(0, 0)");
auto __attribute__((unused)) f2 = [](int, char *) { return 7; };
static_assert(is_callable<decltype(f2), int, char *>::value, "f2(int, char *)");
static_assert(!is_callable<decltype(f2), int, int>::value, "f2(int, int)");
问题不在于 is_callable
测试 class,而是你对它的使用。
在函数的模板参数列表中使用std::enable_if
时,必须这样使用:
template <typename T>
struct runner {
T t;
template
<
typename F,
std::enable_if_t<is_callable<std::decay_t<F>, T&>::value || is_callable<F, T&&>::value>* = nullptr
>
void run(F&& f) {
return f(t);
}
};
您尝试使用的表格用作尾随 return 类型:
template<typename F>
auto run(F&& f)
-> std::enable_if_t<is_callable<std::decay_t<F>, T&>::value || is_callable<F, T&&>::value>
{
return f(t);
}
is_callable
类型正确。好像我之前在 stack overflow 上发过一篇。
我使用的is_callable
结构定义如下
template <typename F, typename... Args>
struct is_callable {
template <typename U>
static auto test(U* p) -> decltype((*p)(std::declval<Args>()...), void(), std::true_type());
template <typename U>
static auto test(...) -> decltype(std::false_type());
static constexpr bool value = decltype(test<F>(nullptr))::value;
};
我正在使用它来测试声明为的 lambda:
template <typename T>
struct runner {
T t;
template <typename F, typename = typename std::enable_if<is_callable<F, T&>::value || is_callable<F, T&&>::value>::type>
void run(F&& f) {
return f(t);
}
};
runner<int> a{0};
a.run([&] (auto& x) {
x++;
});
为什么在 AppleClang 的 enable_if
上编译失败? auto
不应该正确推断吗?
您的 true_type
test
案例看起来不对。无论如何,您的代码比需要的更复杂。尝试以下最小工作示例:
#include <utility>
template<typename F, typename...Args> struct is_callable {
template<typename F2, typename...Args2> static constexpr std::true_type
test(decltype(std::declval<F2>()(std::declval<Args2>()...)) *) { return {}; }
template<typename F2, typename...Args2> static constexpr std::false_type
test(...) { return {}; }
static constexpr bool value = decltype(test<F, Args...>(nullptr))::value;
};
void f0();
static_assert(is_callable<decltype(f0)>::value, "f0()");
static_assert(!is_callable<decltype(f0), int>::value, "f0(0)");
int f1(int);
static_assert(!is_callable<decltype(f1)>::value, "f1()");
static_assert(is_callable<decltype(f1), int>::value, "f1(0)");
static_assert(!is_callable<decltype(f1), int, int>::value, "f1(0, 0)");
auto __attribute__((unused)) f2 = [](int, char *) { return 7; };
static_assert(is_callable<decltype(f2), int, char *>::value, "f2(int, char *)");
static_assert(!is_callable<decltype(f2), int, int>::value, "f2(int, int)");
问题不在于 is_callable
测试 class,而是你对它的使用。
在函数的模板参数列表中使用std::enable_if
时,必须这样使用:
template <typename T>
struct runner {
T t;
template
<
typename F,
std::enable_if_t<is_callable<std::decay_t<F>, T&>::value || is_callable<F, T&&>::value>* = nullptr
>
void run(F&& f) {
return f(t);
}
};
您尝试使用的表格用作尾随 return 类型:
template<typename F>
auto run(F&& f)
-> std::enable_if_t<is_callable<std::decay_t<F>, T&>::value || is_callable<F, T&&>::value>
{
return f(t);
}
is_callable
类型正确。好像我之前在 stack overflow 上发过一篇。