Item 19: Understand the origin of temporary objects.
When programmers speak amongst themselves, they often refer to variables that are needed for only a short while as "temporaries." For example, in this swap
template<class T> void swap(T& object1, T& object2) { T temp = object1; object1 = object2; object2 = temp; }
it's common to call temp
a "temporary." As far as C++ is concerned, however, temp
is not a temporary at all. It's simply an object local to a
True temporary objects in C++ are invisible they don't appear in your source code. They arise whenever a non-heap object is created but not named. Such unnamed objects usually arise in one of two situations: when implicit type conversions are applied to make function calls succeed and when functions return objects. It's important to understand how and why these temporary objects are created and destroyed, because the attendant costs of their construction and destruction can have a noticeable impact on the performance of your
Consider first the case in which temporary objects are created to make function calls succeed. This happens when the type of object passed to a function is not the same as the type of the parameter to which it is being bound. For example, consider a function that counts the number of occurrences of a character in a
// returns the number of occurrences of ch in str size_t countChar(const string& str, char ch); char buffer[MAX_STRING_LEN]; char c; // read in a char and a string; use setw to avoid // overflowing buffer when reading the string cin >> c >> setw(MAX_STRING_LEN) >> buffer; cout << "There are " << countChar(buffer, c) << " occurrences of the character " << c << " in " << buffer << endl;
Look at the call to countChar
. The first argument passed is a char
array, but the corresponding function parameter is of type const
string&
. This call can succeed only if the type mismatch can be eliminated, and your compilers will be happy to eliminate it by creating a temporary object of type string
. That temporary object is initialized by calling the string
constructor with buffer
as its argument. The str
parameter of countChar
is then bound to this temporary string
object. When countChar
returns, the temporary object is automatically
Conversions such as these are convenient (though dangerous see Item 5), but from an efficiency point of view, the construction and destruction of a temporary string
object is an unnecessary expense. There are two general ways to eliminate it. One is to redesign your code so conversions like these can't take place. That strategy is examined in Item 5. An alternative tack is to modify your software so that the conversions are unnecessary. Item 21 describes how you can do
These conversions occur only when passing objects by value or when passing to a reference-to-const
parameter. They do not occur when passing an object to a reference-to-non-const
parameter. Consider this
void uppercasify(string& str); // changes all chars in // str to upper case
In the character-counting example, a char
array could be successfully passed to countChar
, but here, trying to call uppercasify
with a char
array
char subtleBookPlug[] = "Effective C++"; uppercasify(subtleBookPlug); // error!
No temporary is created to make the call succeed. Why
Suppose a temporary were created. Then the temporary would be passed to uppercasify
, which would modify the temporary so its characters were in upper case. But the actual argument to the function call subtleBookPlug
would not be affected; only the temporary string
object generated from subtleBookPlug
would be changed. Surely this is not what the programmer intended. That programmer passed subtleBookPlug
to uppercasify
, and that programmer expected subtleBookPlug
to be modified. Implicit type conversion for references-to-non-const
objects, then, would allow temporary objects to be changed when programmers expected non-temporary objects to be modified. That's why the language prohibits the generation of temporaries for non-const
reference parameters. Reference-to-const
parameters don't suffer from this problem, because such parameters, by virtue of being const
, can't be
The second set of circumstances under which temporary objects are created is when a function returns an object. For instance, operator+
must return an object that represents the sum of its operands (see Item E23). Given a type Number
, for example, operator+
for that type would be declared like
const Number operator+(const Number& lhs, const Number& rhs);
The return value of this function is a temporary, because it has no name: it's just the function's return value. You must pay to construct and destruct this object each time you call operator+
. (For an explanation of why the return value is const
, see Item E21.)
As usual, you don't want to incur this cost. For this particular function, you can avoid paying by switching to a similar function, operator+=
; Item 22 tells you about this transformation. For most functions that return objects, however, switching to a different function is not an option and there is no way to avoid the construction and destruction of the return value. At least, there's no way to avoid it conceptually. Between concept and reality, however, lies a murky zone called optimization, and sometimes you can write your object-returning functions in a way that allows your compilers to optimize temporary objects out of existence. Of these optimizations, the most common and useful is the return value optimization, which is the subject of Item 20.
The bottom line is that temporary objects can be costly, so you want to eliminate them whenever you can. More important than this, however, is to train yourself to look for places where temporary objects may be created. Anytime you see a reference-to-const
parameter, the possibility exists that a temporary will be created to bind to that parameter. Anytime you see a function returning an object, a temporary will be created (and later destroyed). Learn to look for such constructs, and your insight into the cost of "behind the scenes" compiler actions will markedly