Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

deducing references to const from rvalue arguments

Okay, this may seem like a silly question, but here it goes:

template <typename T>
void foo(T& x)
{
}

int main()
{
    foo(42);
    // error in passing argument 1 of 'void foo(T&) [with T = int]'
}

What is preventing C++ to instantiate the foo function template with T = const int instead?

like image 783
fredoverflow Avatar asked May 20 '11 16:05

fredoverflow


2 Answers

The problem is that template type deduction has to work out an exact match, and in that particular case, because of the reference in the signature, an exact match requires an lvalue. The value 42, is not an lvalue, but rather an rvalue, and resolving T with const int would not yield a perfect match. Since template type deduction is limited to exact matches, that deduction is not allowed.

If instead of using a literal you use a non mutable lvalue, then the compiler will deduce the type appropriatedly, as const int will become a perfect match for the argument:

const int k = 10;
foo( k );            // foo<const int>( const int & ) is a perfect match

Now there is a special rule that enables calling a function that takes a const reference (nonmutable lvalue) with an rvalue, that implies creation of a temporary lvalue which is later bound to the reference, but for that rule to kick in the function has to have that signature before hand, which is why explicitly stating that the type of the template is const int works: foo<const int>(42).

like image 144
David Rodríguez - dribeas Avatar answered Oct 15 '22 13:10

David Rodríguez - dribeas


Them's the rules ;-). If you leave the compiler to deduce the type from the argument, it picks the simplest thing it can.

It doesn't seem unreasonable to me. Your template is saying it expects a non-const reference, so it doesn't compile with an rvalue.

You could either tell it what you mean at the call site: foo<int const>(42); or change your template to make it clear it doesn't need a mutable reference: template <typename T> void foo(T const & x) { }.

In C++11 you have more options for expressing what your template will and will not accept.

like image 1
Alan Stokes Avatar answered Oct 15 '22 15:10

Alan Stokes