Effective C++ 笔记(二)
阅读原文时间:2023年08月23日阅读:1

16、保证异常安全

1 void PrettyMenu::changBackground(std::istream &imgSrc)
2 {
3 lock(&mutex);
4 delete bgImage;
5 ++imageChanges;
6 bgImage = new Image(imgSrc);
7 unlock(&mutex);
8 }

上述代码中的问题,

a)如果 new Image 失败抛出异常会导致 mutex 没有释放。

b)如果 new Image 失败, bgImage 指向了一个已经被释放的内存。

c)如果 new Image 失败, imageChanges +1,这不符合逻辑。

异常安全函数提供三个保证:

a) 基本承诺:如果抛出异常,程序仍然保持在有效状态。没有任何对象或数据结构被破坏。

b)强烈保证:如果抛出异常,程序状态不变。(如果异常抛出,程序状态不改变,如果成功,完全成功,如果失败,完全失败。)

c)不抛掷保证:保证绝不抛异常,总是能完成承诺的功能。

17、inlining 的里里外外

inline 函数背后的整体观念是,将“对此函数调用”以函数本体替换之。这样做可能增加目标码的大小。在一台内存有限的机器上,过度热衷inlining, 会造成程序体积太大(对可用内存空间而言)。inline造成代码膨胀会造成额外的换页行为,降低指令高速缓存的击中率。

inline函数只是对编译器的一个申请,不是强制命令,申请可以是隐喻提出,也可以明确提出。隐喻提出是将函数定义在class定义式内。

构造函数和析构函数往往是inlining 的糟糕候选人。(往往比我们想象的做的更多。)

18、public继承 == is-a 关系

适用于base class 上的每一件事,一定也适用于derived class。因为每个derived class对象也都是一个base class。

19、避免遮掩继承而来的名称

#include

using namespace std;

class Base
{
public:
virtual void mf1() = 0;
virtual void mf1(int){}
virtual void mf2(){}
void mf3(){}
void mf3(double){}
};

class Derived : public Base
{
public:
// using Base::mf1;
// using Base::mf3;

virtual void mf1(){}  
void mf3(){}  
void mf4(){}  

};

int main()
{
cout << "Hello World!" << endl;

Derived d;  
d.mf1();  
d.mf1(5);  
d.mf3();  
d.mf3(2.5);

return 0;  

}

因为Derived实现了mf1(), 所以不在继承 mf1(int). 导致 d.mf1(5); 运行失败。

如果你在使用public继承,而又不继承那些重载函数,就违反了。 base 和 derived 之间的 is-a 关系。

在 Derived 中 声明 使用 using Base::mf1, using Base::mf3 可以解决上面的问题。

20、区分接口继承和实现继承

声明 pure virtual 函数目的是为了让Derived class 只继承函数接口。(pure virtual 可以提供函数定义。调用途径是调用时明确指定出class名称)

声明 impure virtual函数的目的,是让Derived class继承该函数接口和缺省实现。

non-virtual 函数具体指定接口继承以及强制性实现继承。

21、绝不重新定义继承而来的缺省参数值

virtual函数系动态绑定,而缺省参数值却是静态绑定。

#include

using namespace std;

class Shape
{
public:
virtual void draw(int a = 0) {
cout << "a : " << a << endl;
}
};

class Rectangle : public Shape
{
public:
virtual void draw(int a = 1) {
cout << "a : " << a << endl;
}
};

class Cycle : public Shape
{
public:
virtual void draw(int a = 2) {
cout << "a : " << a << endl;
}
};

int main(void)
{
Shape *ps;
Shape *pr = new Rectangle;
Shape *pc = new Cycle;

pr->draw();  
pc->draw();

Rectangle rec;  
rec.draw();

Cycle cyc;  
cyc.draw();

return 0;  

}

22、复合意味着has-a

当两个类之间不适合is-a的关系,所以public继承不适合用来塑模,可以在class A中包含 class B。

23、明智而慎重的使用private继承

私有继承,编译器不会自动的将Derived对象转为Base对象。private继承而来的所有成员,在derived中都变成private。

private继承意味着,只继承部分实现,不继承接口。

如果D以private继承B,意思是D对象根据B对象实现而得,再没有其他含义。

在has-a 和 private继承之间,尽量选择has-a,只有当涉及私有protected成员或者virtual函数牵扯进来时。