Effective C++ 笔记(2)

第二部分: 构造/析构/赋值运算

条款05: 了解C++默默编写并调用哪些函数

对于一个类,如果自己没声明,那么编译器会默认生成一个copy构造函数、一个copy assignment操作符和一个析构函数。此外如果你没有声明任何构造函数,编译器还会声明一个default构造函数。
所有这些函数都是public并且inline的。
因此,如果你写下

class Empty {};

这就好像你写下这样的代码

class Empty {
    Empty() { ... }                             //default构造函数
    Empty(const Empty& rhs){ ... }             //copy构造函数
    ~Empty(){ ... }                            //析构函数

    Empty& operator=(const Empty& rhs){ ... } //copy assignment操作符
}
请记住

编译器可以暗自为class创建default构造函数、copy构造函数、copy assignment操作符以及析构函数。

条款06: 若不想使用编译器自动生成的函数, 就该明确拒绝

如果不想编译器支持某个类的copy构造函数或者copy assignment操作符, 可以自己创建它们的空函数,然后声明为private(一般来说,因为private声明,它们不能被调用,一旦menber函数和friend函数调用,则会生成连接错误阻止它们的行为)

不想让menber函数和friend函数的调用到连接阶段才报错?可以创建一个Uncopyable基类,继承它即可。

class Uncopyable {
protected:
    Uncopyable(){}
    ~Uncopyable(){}
private:
    Uncopyable(const Uncopyable&);
    Uncopyable& operator= (const Uncopyable&);
}
请记住
  • 为驳回编译器自动提供的机能,可将相应的成员函数声明为private并且不予实现。使用像Uncopyable这样的base class也是一种做法。

条款07: 为多态基类声明virtual析构函数

请记住
  • 带多态性质的基类应该声明一个虚析构函数。如果类带有任何虚函数,它就应该拥有一个虚析构函数。
  • 一个类的设计目的如果不是作为基类使用,或不是为了具备多态性,就不该声明虚析构函数。

条款08: 别让异常逃离析构函数

请记住
  • 析构函数绝对不要吐出异常。如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常, 然后吞下它们(不传播)或者结束程序。
  • 如果客户需要对某个操作函数运行期间抛出的异常做出反应,那么class应该提供一个普通函数(而非在析构函数中)执行该操作。

条款09: 绝不在构造和析构过程中调用virtual函数

请记住
  • 在构造和析构期间不要调用virtual函数,因为这类调用从不下降到派生类
struct Super {
    Super() { test(); }
    virtual ~Super() { test(); }
    virtual void test() { std::cerr << "super" << std::endl; }
};

struct Derived : Super {
    void test() override { std::cerr << "derived" << std::endl; }
};

int main() {
    Super *p = new Derived;
    p->test();
    delete p;
}

输出:

super
derived
super

条款10: 令operator=返回一个reference to *this

为了让自定义类型实现如下的连锁赋值

int x, y, z;
x = y = z = 15;

class的赋值操作符必须返回一个reference指向操作符的左侧实参

class Widget{
    public:
    Widget& operator=(const Widget& rhs) { //返回类型是个reference
        ...                                 
        return *this                       //返回左侧对象             
    }
}

这份协议被所有内置类型和标准程序库所提供的类型共同遵守。

请记住:
  • 令赋值操作符返回一个reference to *this

条款11: 在operator=中处理自我赋值

请记住:
  • 确保当对象自我赋值时=有良好行为。其中技术包括比较“来源对象”和“目标对象”的地址、精心周到的语句顺序、以及copy-and-swap。
  • 确定任何函数如果操作一个以上的对象,而其中多个对象是同一个对象时,其行为仍然正确。

条款12: 复制对象时勿忘其每一个成分

为派生类撰写copying函数的重责大任,必须小心地复制其base class 成分。应该让派生类的copying构造函数调用相应的基类函数:

PriorityCustomer::PriorityCustomer(const PriorityCustomer& rhs)
    : Customer(rhs),
      priority(rhs.priority) {
          logCall("PriorityCustomer copy constructor");
      }
PriorityCustomer&
PriorityCustomer::operator=(const PriorityCustomer& rhs) {
    logCall("PriorityCustomer copy assignment operator");
    Customer::operator=(rhs);
    priority = rhs.priority;
    return *this;
}

当编写一个copying函数,请确保(1)复制所有local成员变量,(2)调用基类内的适当copying函数。

请记住:
  • Copying函数应该确保复制"对象内的所有成员变量"以及"所有base class成分"
  • 不要尝试以某个copying函数实现另一个copying函数。应该将共同机能放进第三个函数,并由两个coping函数共同调用。(不应让copy assignment操作符调用copy构造函数,也不该让copy构造函数调用copy assignment操作符)
上一篇:c++自己实现简单智能指针


下一篇:小心,自定义拷贝函数