There are a number of similar questions/answers, but I couldn't quite put those answers together to serve my purposes. I want a traits
template<typename Func, typename ReturnType, typename... Args>
struct returns_a { static const bool value; };
such that
returns_a<F,T,Args>::value
is true if F(Args)
is well formed and returns a T
. After some more research, I got it working as follows:
// 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...> {};
which now works fine for functors and functions. Here is a simple test:
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;
}
Well, this works, but it seems a bit cumbersome. Anybody has better/more elegant/shorter solutions?
I think this will do:-
(Corrected including aschepler's test case)
(Built with clang 3.2, gcc 4.7.2, gcc 4.8.1)
Live example.
A better signature for
will_return
, in my opinion, would be:which gives you:
and
which I think looks prettier.
If you prefer "can be converted" rather than "is the same type", you can change
is_same
above tois_convertible
.Well, this is embarrassing: I found a rather simple (hence elegant) way:
when all the hard work is done in
std::function<>
.