5.了解C++默默编写并调用那些函数
编译器可以暗自为class创建default构造函数(前提是程序员自己不编写任何的构造函数),copy构造函数,copy assignment操作符,以及析构函数。
6.若不想使用编译器自动生成的函数,就该明确拒绝
如果你想要拒绝一些编译器默认为你生成的函数功能,那么最好的方案就是自己声明他们,但是不给他们定义,然后将他们的权限设置为private(这样一般就无法调用因为编译器生成的默认都是public,所以才可以被外部使用),或者你还可以继承一个不支持哪些功能的基类。
7.为多态基类声明virtual析构函数
待解决的问题::::::::::::::::当derived class对象经由一个base class指针被删除,而该base class带着一个non-virtual析构函数,其结果未有定义------实际执行时通常发生的是对象的derived成分没被销毁。
解决方案:给base class一个virtual析构函数,此后删除derived class对象的结果就会如你所料。
但是如果classes的设计目的不是作为base classes使用,或不是为了具备多态性,就不该声明virtual析构函数。
8.别让异常逃离析构函数
原因:假设一个vector v,存在,当它销毁的时候应该销毁其内含的所有对象,但是如果他的对象的析构函数会抛出异常,一个还好,若是抛出两个则对C++而言太多了,程序不是结束执行就是导致不明确行为,这是不容许的。
合理的解决方案应该是重新设计析构接口,自己写一个close函数来进行关闭,若是关闭失败(调用std::abort())那就强迫结束或者吞下异常。(反正绝对不可以抛出异常)
9.绝不在构造和析构过程中调用virtual函数
原因:若是构造或者析构函数中调用了virtual函数,那么在构造或者析构的时候该函数会是base class版本而不会降级。(因为在derived的base class构造期间,对象的类型是base class而不是derived class)。
10.令operator=返回一个reference to* this
Widget& operator=(const Widget& rhs)
{
...
return *this;
}
11.在operator=中处理"自我赋值"
赋值如果遇到了有关动态开辟空间的操作记得语句顺序,愤青来源对象和目标对象的地址。
12.复制对象的时候不要忘记他每一个成分
不要让copy和copy assignment互相调用,哪怕有代码重复的部分,你可以把重复的部分写一个函数让他们去调用。
(1)复制所有local成员变量(2)调用所有base classes内的适当的copying函数