有许多类似的问题/答案,但我无法将这些答案放在一起以达到我的目的。我想要一个特质
template<typename Func, typename ReturnType, typename... Args>
struct returns_a { static const bool value; };
这样
returns_a<F,T,Args>::value
如果F(Args)
格式正确并且返回T
,则为真。经过一些研究,我得到了如下工作:
// value is true if Func(Args...) is well formed
template<typename Func, typename... Args>
class is_callable
{
template <typename F>
static decltype(std::declval<F>()(std::declval<Args>()...), void(), 0) test(int);
template <typename>
static void test(...);
public:
static const bool value = !std::is_void<decltype(test<Func>(0))>::value;
};
// return_type<FunctionSignature>::type is the return type of the Function
template<typename>
struct return_type {};
template<typename ReturnType, typename... Args>
struct return_type<ReturnType(Args...)>
{ typedef ReturnType type; };
// helper class, required to use SFINAE together with variadic templates parameter
// generic case: Func(Args...) is not well-defined
template <typename Func, typename ReturnType, typename dummy, typename... Args>
struct returns_a_helper { static const bool value = false; };
// Func is a function signature
template <typename Func, typename ReturnType, typename... Args>
struct returns_a_helper<Func, ReturnType, typename
std::enable_if<std::is_function<Func>::value>::type, Args...>
{
static const bool value =
std::is_convertible<typename return_type<Func>::type,
ReturnType>::value;
};
// Func(Args...) not a function call, but well-defined
template <typename Func, typename ReturnType, typename... Args>
struct returns_a_helper<Func,ReturnType,typename
std::enable_if<is_callable<Func>::value &&
!std::is_function<Func>::value
>::type, Args...>
{
static const bool value =
std::is_convertible<typename std::result_of<Func(Args...)>::type,
ReturnType>::value;
};
template <typename Func, typename ReturnType, typename... Args>
struct returns_a : returns_a_helper<Func, ReturnType, void, Args...> {};
现在适用于仿函数和函数。这是一个简单的测试:
struct base { virtual bool member(int) const = 0; };
struct foo : base { bool member(int x) const { return x&2; } };
struct bar { foo operator()() { return foo(); } };
foo free_function() { return foo(); }
template<typename T, typename Func>
void test(Func const&func)
{
std::cout << std::boolalpha << returns_a<Func,T>::value << std::endl;
}
int main()
{
foo x;
bar m;
test<const base&>([&]() { return x; });
test<const base&>(m);
test<const base&>(free_function);
return 0;
}
嗯,这很有效,但看起来有点麻烦。任何人都有更好/更优雅/更短的解决方案?
答案 0 :(得分:1)
#include <tuple>
#include <utility>
template<typename Func, typename R, typename Args, typename=void>
struct will_return_helper: std::false_type {};
template<typename Func, typename R, typename... Args>
struct will_return_helper<
Func, R, std::tuple<Args...>,
typename std::enable_if<
std::is_same<
R,
decltype( std::declval<Func&>()( std::declval<Args>()... ) )
>::value
>::type
> : std::true_type {};
template<typename Func, typename R, typename... Args>
struct will_return:
will_return_helper< typename std::decay<Func>::type, R, std::tuple<Args...> >
{};
#include <iostream>
struct Foo {
int operator()(double) {return 0;}
};
int main()
{
std::cout << "1 = "<< will_return< int(), int >::value << "\n";
std::cout << "1 = "<< will_return< int(*)(), int >::value << "\n";
std::cout << "0 = "<< will_return< int(*)(), double >::value << "\n";
std::cout << "1 = "<< will_return< Foo, int, double >::value << "\n";
std::cout << "1 = "<< will_return< Foo, int, int >::value << "\n";
std::cout << "0 = "<< will_return< Foo, double, int >::value << "\n";
}
在我看来,will_return
更好的签名是:
template<typename Func, typename Sig>
struct will_return;
template<typename Func, typename R, typename... Args>
struct will_return<Func, R(Args...)>:
will_return_helper< typename std::decay<Func>::type, R, std::tuple<Args...> >
{};
给你:
std::cout << "1 = "<< will_return< int(), int() >::value << "\n";
和
std::cout << "1 = "<< will_return< Foo, int(double) >::value << "\n";
我认为看起来更漂亮。
如果您更喜欢“可以转换”而不是“属于同一类型”,则可以将上面的is_same
更改为is_convertible
。
答案 1 :(得分:1)
我认为这样做: -
(更正包括aschepler的测试用例)
#include <type_traits>
template<typename Func, typename Ret, typename... Args>
struct returns_a
{
template<typename F, typename ...As>
static constexpr bool test(
decltype(std::declval<F>()(std::declval<As>()...)) * prt) {
return std::is_same<Ret *,decltype(prt)>::value;
}
template <typename F, typename ...As>
static constexpr bool test(...) {
return false;
}
static const bool value = test<Func,Args...>(static_cast<Ret *>(0));
};
// Testing...
#include <iostream>
void fn0();
int fn1(int);
int fn2(int,int);
struct cls{};
struct fntor
{
int operator()(int i) { return 1; }
};
auto lamb0 = [](int i) -> int { return i; };
struct B {};
struct D : public B {};
auto lamb1 = []{ return B{}; };
int main()
{
std::cout << returns_a<decltype(fn0),void>::value << std::endl; // 1
std::cout << returns_a<decltype(fn1),int,int>::value << std::endl; // 1
std::cout << returns_a<decltype(fn1),int,double>::value << std::endl; // 1
std::cout << returns_a<decltype(fn1),double,int>::value << std::endl; // 0
std::cout << returns_a<decltype(fn1),char,int>::value << std::endl; // 0
std::cout << returns_a<decltype(fn1),unsigned,int>::value << std::endl; // 0
std::cout << returns_a<decltype(fn2),int,int,int>::value << std::endl; // 1
std::cout << returns_a<decltype(fn2),int,char,float>::value << std::endl; // 1
std::cout << returns_a<cls,int,int>::value << std::endl; // 0
std::cout << returns_a<fntor,int,int>::value << std::endl; // 1
std::cout << returns_a<decltype(lamb0),int,int>::value << std::endl; // 1
std::cout << returns_a<double,int,int>::value << std::endl; // 0
std::cout << returns_a<decltype(lamb1), D>::value << std::endl; //0
return 0;
}
(使用clang 3.2,gcc 4.7.2,gcc 4.8.1构建)
答案 2 :(得分:0)
嗯,这很令人尴尬:我发现了一种相当简单(因此优雅)的方式:
template <typename Func, typename ReturnType, typename... Args>
using returns_a = std::is_convertible<Func, std::function<ReturnType(Args...)>>;
在std::function<>
完成所有艰苦工作。