0%

Chapter-12-compound-types-references-pointers

# 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
2
3
4
5
// regular types
int // a normal int type (not an reference)
int& // an lvalue reference to an int object
double& // an lvalue reference to a double object
const int& // an lvalue reference to a const int object

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
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <iostream>

int main()
{
int x { 5 };

{
int& ref { x }; // ref is a reference to x
std::cout << ref << '\n'; // prints value of ref (5)
} // ref is destroyed here -- x is unaware of this

std::cout << x << '\n'; // prints value of x (5)

return 0;
} // x destroyed here

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
2
3
4
5
6
7
8
9
10
11
12
#include <iostream>

int main()
{
const int x { 5 }; // x is a non-modifiable lvalue
const int& ref { x }; // okay: ref is a an lvalue reference to a const value

std::cout << ref << '\n'; // okay: we can access the const object
ref = 6; // error: we can not modify an object through a const reference

return 0;
}

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
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
#include <iostream>
#include <string>

const std::string& getProgramName() // returns a const reference
{
static const std::string s_programName { "Calculator" }; // has static duration, destroyed at end of program

return s_programName;
}

int main()
{
std::cout << "This program is named " << getProgramName();

return 0;
}

It is ok, because s_programName has static duration, s_programName will exist until the end of the program.