I did do some research on std::result_of<>
, and am aware how it's used on a high level, but am still pretty confused on this magical thing.
So, if I understand it correctly, the form R (Args..)
should be interpreted by the compiler as a function type taking the arguments of types Args...
and returning type R
, and the usage is very much justified when specifying std::function<int(int)>
, which declares a function type taking int
as its only argument, and returning int
.
However, the meaning of the above suddenly changes when the same notion is used in the context of std::result_of<F(Args...)>
(note that I changed R
to F
, and removed the (space) between
F
and (
to signify the difference - according to cppreference, the meaning of F(Args...)
becomes a compile time INVOKE
expression - I mean, I kind of understand that the main purpose of std::result_of<F(Args...)>
is to use the notion of INVOKE
to select the correct overload (if more than one overloads are present), and of course deduce the return type of such invocation (during compile time resolution) - effectively, this would roughly be similar to: decltype(std::declval<F>()(std::declval<Args>()...))
- however, the notion of F (Args...)
- comparing to the std::function<R (Args...)>
case - would have been interpreted by the compiler as a function type taking Args...
and returning the type F
, where now F
is the function type that we are trying to get the return type of!
Of course, with the help of decltype
and std::declval
, the implementation of std::result_of
seems reasonable, but I read that boost somehow implemented result_of
in C++98/03 where decltype
and std::declval
were not existing - I'm wondering how would that be possible (maybe with some very hack'ish tricks)?
So to re-iterate: is the notion of R (Args...)
in the context of being a template argument always interpreted / deduced as a function type with a return type R
, regardless if the enclosing template is std::function<>
or std::result_of<>
? while std::result_of<>
somehow does some kind of reinterpretation of the "return type" F
, and "returns" the actual return type as defined by std::result_of<F(Args...)>::type
? Or std::result_of<F(Args...)>
just interprets F(Args...)
differently, so it knows it's a magical INVOKE
expression?
Thanks for your upcoming clarification!
To implement it without
decltype
—well, a simple carcass oldschool implementation that covers few basic cases would probably be something likeSomewhat more sophisticated dispatch is required for
operator()
case, but otherwise—this is the approximate way to go.