Re: l-values and r-values
On Dec 13, 10:33 pm, "Joe Smith" <unknown_kev_...@hotmail.com> wrote:
"Taras_96" <taras...@gmail.com> wrote in message
news:8e5d755a-5c2f-493e-b443-d1d41112fc6b@m16g2000vbp.googlegroups.com...
I think I mostly understand l-values and r-values after
reading a few archived posts on this board:
My attempt to sumarize conclusions made so-far:
All l-values are objects in memory, and therefore have an
address. R-values may or may not be values in memory, and may
or may not have an address.
More precisely, r-values are objects in memory if and only if
they have class type. Of course, there is no way a conforming
program can tell if a non-class type rvalue is an object
(occupies memory) or not. And if you use the rvalue to
initialize a reference, it must occupy memory. So there's
really no reason to make the distinction---the standard could
just as easily say that all values are objects.
R-values can be converted to lvalues thanks to the const T&
binding.
Except that it's not considered a conversion, according to the
standard. Similarly, if you call a member function on an
r-value, the compiler has to give it an address, in order to
initialize the this pointer.
L-values can be converted to r-values as needed.
Temporaries are normally R-values but because of the r-value
to const T& binding, they can be converted to L-values.
That's not quite the wording in the standard, but I think it
corresponds rather closely to the reality.
An important point here is that an object has a lifetime; a
temporary object normally has a lifetime until the end of the
full expression. (There's an interesting point here: does this
include the rvalue used to initialize a reference. In other
words, does the following code contain undefined behavior or
not:
#include <iostream>
void
f( int const& i )
{
pi = &i ;
}
void
g( int i )
{
std::cout << *pi << ',' << i << std::endl ;
}
int
main()
{
f( 1 ), g( 2 ) ;
return 0 ;
}
? If not, it's yet another example of a case where a non-class
rvalue behaves exactly like an object.)
Only lvalues can be on the left hand side of built in
operators that modify the right-hand side, or on wither side
of operator++ and operator--. That does not hold true for
user-defined operator overloads.
In general, only an lvalue can be "modified"; any operator which
modifies something requires an lvalue. What can happen,
especially with class types, is that the rvalue-ness is lost;
the lvalue- or rvalue-ness is not a characteristic of the
object, but of the expression.
L-values may or may not be modifiable. R-values may or may
not be modifiable, due to mutating member functions, or the
use of const_cast with the const T& binding.
Yeah, I'm going to say that this is a mess.
History. There was some discussion about dropping the
lvalue-rvalue distinction completely when the standard was being
formulated, but in the end, it was decided that built-in types
should follow the same rules as they do in C. And while I think
this does make things considerably more complicated, I can
understand that desire.
Anyway, there are a few simple rules which make it easy to
handle:
1. The distinction lvalue/rvalue only applies to expressions.
The standard specifies which expressions result in lvalues,
and it specifies which expressions require lvalues, for
which operands. And that's the only real meaning of
lvalue/rvalue---it's a more or less arbitrary distinction
based on the expression.
2. When considering expressions (and not just for
lvalue-/rvalue-ness), you have to first perform overload
resolution, and replace user defined operators with the
corresponding function call.
--
James Kanze (GABI Software) email:james.kanze@gmail.com
Conseils en informatique orient=E9e objet/
Beratung in objektorientierter Datenverarbeitung
9 place S=E9mard, 78210 St.-Cyr-l'=C9cole, France, +33 (0)1 30 23 00 34