Effective C++ 3nd 读书摘要(一、让自己习惯C++ ; 二、构造,析构,赋值运算)Item1 - 12

一、让自己习惯C++
Item1. 视C++为一个语言联邦
C++由以下部分组成:C、Object-Oriented C++、Template C++、STL

 

Item2. 尽量以const、enum、inline代替#define

 

Item3. 尽可能使用const
1.一共有这样几种const用法:
char greeting[] = "Hello";
char *p = greeting;                    // non-const pointer, non-const data
const char *p = greeting;              // non-const pointer, const data
char * const p = greeting;             // const pointer, non-const data
const char * const p = greeting;       // const pointer, const data

 

另外,以下两种形式是一样的:
void f1(const Widget *pw);         // f1 takes a pointer to a constant Widget object
void f2(Widget const *pw);         // so does f2

 

迭代器也有const:
const std::vector<int>::iterator iter =     // iter acts like a T* const
std::vector<int>::const_iterator cIter =   //cIter acts like a const T*

 

2. 让non-const函数调用const函数以避免重复:

 

const函数可以使得返回的const指针、const引用等等均可以调用一些方法(const对象只能调用const函数)

 

Item4. 确保对象在使用前已经先被初始化
重要的是不能混淆赋值(assignment)与初始化(initialization),通常后者的效率更高,初始化通过成员初值列表。最好以申明次序作为成员列表中的次序。

 

Singleton模式的一个应用:定义于不同编译单元的non-local static对象的初始化次序

即以local static对象代替non-local static对象

 

二、构造、析构、赋值运算
Item5. 了解C++默默编写并调用了哪些函数
这些是C++会默认生成的(如果你没有定义):copy构造、copy assignment、析构(non-virtual)、default构造,所有这些都是public而inline的。

注:上面的析构(non-virtual),除非其base class是虚析构

 

Item6. 若不想使用编译器自动生成的函数,就该明确拒绝
接Item5所言,要想拒绝之:

也可以建一个base class,这样还可以把连接期错误移到编译期:

我们只要继承自此类:

class HomeForSale: private Uncopyable {     // class no longer
  ...                                         // declares copy ctor or copy assign. operator
};

 

Item07. 为多态基类声明virtual析构函数
否则当用base class pointer来delete那个derived class对象时会造成局部销毁。

如果class带有任何virtual函数,base class就应该带有一个virtual析构函数。否则不该用。

 

Item08. 别让异常逃离析构函数
如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常,然后吞下它们(不传播)或结

束程序。
较好的一种做法是:

给客户一次调用close处理错误的机会。

 

Item09. 绝不在构造和析构过程中调用virtual函数
因为在base class构造期间对象的类型还是base class而不是derived classe(此时的virtual函数不是virtual函数,而只是个base class的函数),derived class成分尚为未定义值。


同样,一旦derived class析构函数开始执行,对象内的derived class成员变量便呈现未定义值,那么当进入

base class析构函数后对象就成为一个base class对象。

 

一种弥补方法是令derived classes将必要的构造信息向上传递至base class构造函数。

 

Item10. 令operator=返回一个reference to *this
这样可以支持x=y=z=15这样的赋值连锁形式。

 

Item11. 在operator=中处理“自我赋值”
其中可以用到的技术有:
1. 比较“来源对象”和“目标对象”的地址 if(this == & rhs) return * this;
2. 精心周到的语句顺序

3. copy-and-swap

 

Item12. 复制对象时勿忘其每一个成分
1.要确保(1)复制所有local成员变量;(2)调用所有base classes内的适当的copying函数(copy constructor或copy assignment)


2.不要尝试以一个copying函数实现另一个copying函数,而应该将共同功能放进第三个函数中。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值