C++新特性 强制转换 dynamic_cast
dynamic_cast
用于具有虚函数的基类与派生类之间的指针或引用
- 基类必须具备虚函数
- 原因:dynamic_cast 是运行时类型检查,需要运行时类型信息(RTTI),而这个信息是存储与类的虚函数表关系紧密,只有一个类定义了虚函数,才会有虚函数表
- 运行时检查,转型不成功则返回一个空指针
- 非必要不要使用 dynamic_cast, 有额外的函数开销
常见的转换方式:
- 基类指针或引用转派生类指针(必须使用dynamic_cast)
- 派生类指针或引用转基类指针(可以使用 dynamic_cast, 但是更推荐使用 static_cast)
#include <iostream>
using namespace std;
// 基类与派生类之间的转换
class CFather {
public:
CFather() {
}
virtual void foo() {
std::cout << "CFather()::void foo()" << std::endl;
}
int m_nTest;
};
class CSon : public CFather {
public:
virtual void foo() {
std::cout << "CSon::void foo()" << std::endl;
}
int m_nSon;
};
int main() {
CFather f;
CSon s;
CFather* pFather = &f;
CSon* pSon = &s;
// 使用 dynamic 前提
// pFather = static_cast<CFather*>(pSon);
// 向下转换 父类转子类 不安全
// 有一种语法能检测出这种转换是不安全的, dynamic_cast
// 在运行时刻检测转换是否安全
// pSon = static_cast<CSon*>(pFather);
// pSon->m_nSon = 123;
// dynamic_cast 能够在运行的时刻,检测出被转换的指针的类型(依赖RTTI)
// 有额外的开销,一般而言只有在向下转换时才必须使用
pSon = dynamic_cast<CSon*>(pFather);
if (pSon != nullptr) {
pSon->m_nSon = 123;
std::cout << pSon->m_nSon << std::endl;
}
if (pSon == nullptr) std::cout << "null" << std::endl;
// 向上转换 子类转父类 安全
pFather = static_cast<CFather*>(pSon);
// 具有多态类类类型的向下转换时必须使用,其余情况可以不用
}