RTTI通过typeid和dynamic_cast实现运行时类型识别,用于多态类的类型查询与安全向下转型,但存在性能开销且仅适用于含虚函数的类,建议优先使用虚函数而非类型分支。

在C++中,运行时类型识别(RTTI,Runtime Type Information)是一种允许程序在运行时查询对象实际类型的技术。它主要通过 typeid 和 dynamic_cast 两个关键字实现,适用于多态类(即含有虚函数的类)。下面详细介绍RTTI的核心用法和注意事项。
1. typeid操作符:获取类型信息
typeid 可以返回一个常量引用指向 std::type_info 对象,该对象包含类型的名称和其他信息。使用前需包含头文件
示例:
#include
#include
class Base {
virtual ~Base() {} // 必须有虚函数才能启用RTTI
};
class Derived : public Base {};
int main() {
Base* ptr = new Derived();
std::cout << "指针类型: " << typeid(ptr).name() << std::endl; // 输出指针本身的类型
std::cout << "所指对象类型: " << typeid(*ptr).name() << std::endl; // 输出实际对象类型
delete ptr;
return 0;
}
立即学习“C++免费学习笔记(深入)”;
输出结果可能为:
Derived 类型名取决于编译器(如 GNU C++ 中可能是 7Derived)。要获得可读性好的名字,可以配合 cxxabi.h 使用 __cxa_demangle,但这不是标准C++的一部分。
2. dynamic_cast:安全的向下转型
dynamic_cast 主要用于在继承层次中进行安全的向下转型(downcasting),即把基类指针或引用转换为派生类指针或引用。它依赖RTTI来判断转换是否合法。
如果转换失败:
- 对于指针类型,返回 nullptr
- 对于引用类型,抛出 std::bad_cast 异常
示例:
Base* basePtr = new Derived();
Derived* derivedPtr = dynamic_cast(basePtr);
if (derivedPtr) {
std::cout << "转换成功!" << std::endl;
} else {
std::cout << "转换失败!" << std::endl;
}
注意:只有多态类型(带虚函数)才能使用 dynamic_cast,否则编译会报错。
3. RTTI的限制与性能考虑
RTTI虽然有用,但也有局限性和开销:
- 必须开启虚函数机制,仅对多态类型有效
- 增加二进制体积(每个类保存类型信息)
- 运行时有轻微性能损耗(类型检查需要时间)
- 某些嵌入式环境或高性能场景会禁用RTTI(如使用 -fno-rtti 编译选项)
建议:除非确实需要运行时判断类型,否则优先使用虚函数实现多态行为,避免滥用RTTI。
4. 实际应用场景举例
常见用途包括:
- 调试时打印对象真实类型
- 序列化系统中根据类型选择处理逻辑
- 插件系统或对象工厂中做类型校验
- 事件处理中区分不同类型的消息对象
例如:
void handleObject(Base& obj) {
if (typeid(obj) == typeid(Derived)) {
std::cout << "处理Derived对象\n";
} else if (typeid(obj) == typeid(AnotherDerived)) {
std::cout << "处理AnotherDerived对象\n";
}
}
但更推荐用虚函数代替这种“类型分支”逻辑。
基本上就这些。RTTI是C++中一个实用但应谨慎使用的特性,理解其原理和代价有助于写出更健壮高效的代码。










