Why definition of std::function<>::operator() in the C++ standard is:
R operator()(ArgTypes...) const;
and not
R operator()(ArgTypes&&...) const;
?
One would think that to correctly forward parameters, we need the && and then use std::forward<ArgTypes>...
in the function body when forwarding the call?
I partially reimplemented std::function to test this and I found out that if I use the &&, I get "cannot bind 'xxx' lvalue to 'xxx&&'" from g++ when I try later to pass parameters by value to operator(). I thought that I got enough grasp of the rvalue/forwarding concepts, but still I cannot grok this point. What am I missing?
Perfect forwarding only works when the function itself (in this case
operator()
) is templated and the template arguments are deduced. Forstd::function
, you get theoperator()
argument types from the template parameters of the class itself, which means they'll never be deduced from any arguments.The whole trick behind perfect forwarding is the template argument deduction part, which, together with reference collapsing, is what perfect forwarding is.
I'll just conveniently link to my other answer about
std::forward
here, where I explain how perfect forwarding (andstd::forward
) works.Note that
std::function
'soperator()
doesn't need perfect forwarding, since the user himself decides what the parameters should be. This is also the reason why you cannot just add&&
tooperator()
; take this example: