《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。

相关推荐
岩中竹25 分钟前
力扣热题100—滑动窗口(c++)
数据结构·c++·算法·leetcode
originalHSL26 分钟前
同步/异步日志系统
c++
LIUDAN'S WORLD28 分钟前
C++零基础实践教程 函数 数组、字符串与 Vector
开发语言·c++·算法
ChoSeitaku31 分钟前
NO.95十六届蓝桥杯备战|图论基础-单源最短路|负环|BF判断负环|SPFA判断负环|邮递员送信|采购特价产品|拉近距离|最短路计数(C++)
c++·蓝桥杯·图论
纪元A梦1 小时前
华为OD机试真题——跳格子3(2025A卷:200分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
java·javascript·c++·python·华为od·go·华为od机试题
@hdd1 小时前
C++ | 可变模板参数
c++
BC橡木3 小时前
调度算法
c++
努力努力再努力wz3 小时前
【Linux实践系列】:匿名管道收尾+完善shell外壳程序
linux·运维·服务器·c++