《Effective C++》条款27

尽量少做转型动作

cpp 复制代码
class A
{
public:
	A(int x) :a(x) {};
	virtual void add()
	{
		a++;
	}
private:
	int a;
};
class B :public A
{
public:
	B(int x) :b(x),A(x) {};
	virtual void add()
	{
		static_cast<A>(*this).add();
		b++;
	}

private:
	int b;
};

如上描述把子类转型为A类,调用了A::add()。但是它调用的不是当前对象上的函数,而是稍早转型动作所建立的一个"*this对象之base class成分"的暂时副本上的add 。(成员函数只有一份,关键在于成员函数都有一个this指针,会因此影响成员函数操作的数据。)如果add修改了当前对象的内容,当前对象其实没有被改动,被改动的是副本。然而B类里面的add如果也修改对象,那么该对象会被改变。这使当前对象进入一种"伤残"状态:其base class成分的更改没有落实,而derived class成分的更改倒是落实了。

所以请这么写:

cpp 复制代码
class B :public A
{
public:
	B(int x) :b(x),A(x) {};
	virtual void add()
	{
		A::add();
		b++;
	}

private:
	int b;
};

如果想使用dynamic_cast,通常是你想在一个你认定为子类对象身上执行子类操作函数但是你手上只有一个指向基类的指针或引用。你只能靠它们来处理对象。有两个做法可以避免这个问题:

假设子类有一个唯一的函数func

cpp 复制代码
class B :public A
{
public:
	...
	void func();
	...
};

你可以使用容器并在其中存储直接指向子类对象的指针:

cpp 复制代码
typedef vector<shared_ptr<B>> Bptr;
Bptr bp;
...
for (Bptr::iterator iter = bp.begin();iter != bp.end();++iter)
{
	(*iter)->func();
}

而另一种做法可以让你通过基类接口处理"所有可能之各种A派生类",那就是在基类内提供virtual函数做到你想对任何派生类做的事:

cpp 复制代码
class A
{
public:
	virtual void func()
	{
		// 什么也不做,缺省
	}
	...
};
class B :public A
{
public:
	virtual void func()
	{
		...
	}
	...
};
int main()
{
	typedef vector<shared_ptr<A>> Aptr;
	Aptr ap;
	...
	for (Aptr::iterator iter = ap.begin();iter != ap.end();++iter)
	{
		(*iter)->func();
	}
}

这两种做法都可以替代dynamic_cast。

相关推荐
林开落L11 分钟前
前缀和算法习题篇(上)
c++·算法·leetcode
Prejudices24 分钟前
C++如何调用Python脚本
开发语言·c++·python
单音GG26 分钟前
推荐一个基于协程的C++(lua)游戏服务器
服务器·c++·游戏·lua
qing_04060343 分钟前
C++——多态
开发语言·c++·多态
孙同学_43 分钟前
【C++】—掌握STL vector 类:“Vector简介:动态数组的高效应用”
开发语言·c++
charlie1145141911 小时前
Qt Event事件系统小探2
c++·qt·拖放·事件系统
iiiiiankor1 小时前
C/C++内存管理 | new的机制 | 重载自己的operator new
java·c语言·c++
小辛学西嘎嘎1 小时前
C/C++精品项目之图床共享云存储(3):网络缓冲区类和main
c语言·开发语言·c++
c语言鹌鹑蛋2 小时前
C++初阶 --- 类和对象(1)
开发语言·c++
Jack黄从零学c++2 小时前
opencv(c++)图像的灰度转换
c++·人工智能·opencv