Item 5 - Know what functions C++ silently writes and 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

class Empty{};

it's essentially the same as if you'd written this:

class Empty {
public:
    Empty(){...}                                // default constructor
    Empty(const Empty& rhs){...}                // copy constructor
    ~Empty(){...}                               // destructor — see below for whether it's virtual
    Empty& operator=(const Empty& rhs){...}     // copy assignment operator
};

These functions are generated only if they are needed. The following code will cause each function to be generated:

Empty e1;                    // default constructor;
                             // destructor
Empty e2(e1);                // copy constructor
e2=e1;                       // copy assignment operator

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:

template<typename T>
class NamedObject {
public:
    NamedObject(const char* name, const T& value);
    NamedObject(const std::string& name, const T& value);
    ...
private:
    std::string nameValue;
    T objectValue;
};

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:

NamedObject<int> no1("Smallest prime Number",2);
NamedObject<int> no2(no1);                            // calls 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:

template<typename T>
class NamedObject{
public:
    // this ctor no longer takes aconst name, because nameValue
    // is now a reference-to-non-const string. The char* constructor
    // is gone, because we must have a strng to refer to.
    NamedObject(std::string& name, const T& value);
    ...                        // as above, assume no operator= is decalared
private:
    std::string& nameValue;    // this is now a reference
    const T objectValue;       // this is now const
};

Now consider what should happen here:

std::string newDog("Persephone");
std::string oldDog("Satch");

NamedObject<int> p(newDog, 2);         // when I originally wrote this, our dog Persephone was about to have her second birthday
NamedObject<int> s(oldDog, 36);        // the family dog Satch (from my childhood) would be 36 if she were still alive
p=s;                                   // what should happed to the data members in p?

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.

results matching ""

    No results matching ""