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

相关推荐
灯厂码农10 分钟前
C++文件操作
开发语言·c++
️停云️26 分钟前
C++异常与智能指针
开发语言·c++
chenyuhao20241 小时前
MySQL事务
开发语言·数据库·c++·后端·mysql
爪哇部落算法小助手1 小时前
每日两题day59
数据结构·c++·算法
D_evil__1 小时前
[C++高频精进] 现代C++特性:右值引用和移动语义
c++
Mr_WangAndy1 小时前
C++14新特性_第一章C++语言特性_Lambda初始化捕获,decltype(auto)
c++·c++40周年·lambda初始化捕获·decltype auto
不会c嘎嘎2 小时前
【C++】深入理解多态:从用法到原理
开发语言·c++
REDcker2 小时前
软件开发者需要关注CPU指令集差异吗?
linux·c++·操作系统·c·cpu·指令集·加密算法
不知所云,2 小时前
5. SDL3 库项目引入
c++·sdl3
C++ 老炮儿的技术栈2 小时前
用密码学安全随机数生成256位密钥
c语言·开发语言·c++·windows·安全·密码学·visual studio