# Value categories
lvalue and rvalue
An lvalue (pronounced “ell-value”, short for “left value” or “locator value”, and sometimes written as “l-value”) is an expression that evaluates to an identifiable object or function (or bit-field). An rvalue (pronounced “arr-value”, short for “right value”, and sometimes written as r-value
) is an expression that is not an lvalue. Rvalue expressions evaluate to a value.
Lvalue expressions evaluate to an identifiable object.
Rvalue expressions evaluate to a value.
Lvalue to rvalue conversion
Lvalue expressions will implicitly convert to rvalue expressions in contexts where an rvalue is expected but an lvalue is provided.
Lvalue references
1 | // regular types |
Reference scope and duration
Reference variables follow the same scoping and duration rules that normal variables do.
References and referents have independent lifetimes
- A reference can be destroyed before the object it is referencing.
- The object being referenced can be destroyed before the reference.
1 |
|
Dangling references
When an object being referenced is destroyed before a reference to it, the reference is left referencing an object that no longer exists. Such a reference is called a dangling reference.
References aren’t objects
Perhaps surprisingly, references are not objects in C++. A reference is not required to exist or occupy storage. If possible, the compiler will optimize references away by replacing all occurrences of a reference with the referent.
Because references aren’t objects, they can’t be used anywhere an object is required (e.g. you can’t have a reference to a reference, since an lvalue reference must reference an identifiable object).
Lvalue references to const
1 |
|
Const references bound to temporary objects extend the lifetime of the temporary object
When a const lvalue reference is directly bound to a temporary object, the lifetime of the temporary object is extended to match the lifetime of the reference.
Constexpr lvalue references
Constexpr references have a particular limitation: they can only be bound to objects with static duration (either globals or static locals). This is because the compiler knows where static objects will be instantiated in memory, so it can treat that address as a compile-time constant.
Pass by lvalue reference
Pass by const lvalue reference
Return by reference
The object being returned by reference must exist after the function returns.
eg:
1 |
|
It is ok, because s_programName
has static duration, s_programName
will exist until the end of the program.