Item 5 - Know what functions C++ silently writes a calls.
Empty class is not an empty class. If you don't declare them yourself, compilers will declare their own versions of a copy contructor, a copy assignment operator, and a destructor. Furthermore, if you declare no constructors at all, compilers will also declare a default constructor for you. All these functions will be both public and inline. As a result if you write
it's essentially the same as if you'd written this:
These functions are generated only if they are needed. The following code will cause each function to be generated:
Note: The generated destructor is non-virtual unless it's for a class inheriting from a base class that itself declares a virtual destructor (in which case the function's virtualness comes from the base class).
As for the copy constructor and copy assignment operator, the compiler -generated versions simply copy each non-static data member of the source object over to the target object. #####Example: Consider a NamedObject
template that allows to associate names with objects of type T
:
Because a constructor is declared in NamedObject
, compilers won't generate a default constructor. But NamedObject
declares neither copy constructor nor copy assignment operator, so compilers will generate those functions (if needed). Look, then, at this use of copy constructor:
The compiler-generated copy assignment operator usually behave coorect when the resulting copde is both legal and has a reasonable chance of making sense. If either of these tests fails, compilers will refuse to generate an operator=
for your class. #####Example: Suppose NamedObject
were defined like this, where nameValue
is a reference to a string and objectValue
is a _const_T:
Now consider what should happen here:
Before the assignment, both p.nameValue
and s.nameValue
refer to string
objects, though not the same ones. How should assignment affect p.nameValue
? After the assignment, should p.nameValue
refer to the string
referred to by s.nameValue
, i.e., should the reference itself be modified? If so, that breaks new ground, because
C++ doesn't provide a way to make a reference refer to a different object.
Alternatively, should the string object to which p.nameValue
refers be modified, thus affecting other objects that hold pointers or references to that string
, i.e., objects not directly involved in the assignment? Is that what the compiler-generated copy assignment operator should do?
Because of this, C++ refuses to compile the code. If you want to support copy assignment in a class containing a reference member, you must define the copy assignment operator yourself. Compilers behave similarly for classes containing const
members (such as objectValue
in the modified class above). It's not legal to modify const
members, so compilers are unsure how to treat them during an implicitly generated assignment function. Finally, compilers reject implicit copy assignment operators in derived classes that inherit from base classes declaring the copy assignment operator private. After all, compiler-generated copy assignment operators for derived classes are supposed to handle base class parts, too, but in doing so, they certainly can't invoke member functions the derived class has no right to call.
Things to Remember:
Compilers may implicitly generate a class's default constructor, copy constructor, copy assignment operator, and destructor.
Last updated