RTTI通过typeid和dynamic_cast实现运行时类型识别,需类含虚函数;typeid获取类型信息,dynamic_cast用于安全向下转型,两者均依赖多态且有性能开销,编译器可能禁用,typeid.name()需解构为可读名。

在c++中,运行时类型识别(RTTI, Run-Time Type Information)允许程序在运行时查询对象的实际类型。这主要通过 typeid 操作符和 dynamic_cast 来实现。要使用RTTI,类必须包含虚函数(即多态类型),否则部分功能将受限。
1. 使用 typeid 获取类型信息
typeid 可以返回一个 const std::type_info& 引用,表示对象或类型的运行时类型信息。需要包含头文件 <typeinfo>。
示例:
#include <iostream>
#include <typeinfo>
class Base {
virtual ~Base() {} // 必须有虚函数才能启用RTTI
};
class Derived : public Base {};
int main() {
Derived d;
Base* ptr = &d;
std::cout << “实际类型: ” << typeid(*ptr).name() << std::endl;
if (typeid(*ptr) == typeid(Derived)) {
std::cout << “指针指向的是 Derived 对象” << std::endl;
}
}
注意:typeid(*ptr) 中的 *ptr 是解引用指针,这样才能触发多态行为获取真实类型;如果写成 typeid(ptr),得到的是 Base* 类型本身。
立即学习“C++免费学习笔记(深入)”;
2. 使用 dynamic_cast 进行安全的向下转型
dynamic_cast 用于在继承层次中进行安全的类型转换,尤其是从基类指针转为派生类指针。如果转换失败(类型不匹配),返回 nullptr(对于指针)或抛出异常(对于引用)。
示例:
Base* ptr = new Derived();
Derived* dp = dynamic_cast<Derived*>(ptr);
if (dp) {
std::cout << “转换成功,对象是 Derived 类型” << std::endl;
} else {
std::cout << “转换失败” << std::endl;
}
对于多个继承层级也适用,只要类体系中有虚函数且开启RTTI即可。
3. RTTI 的限制与注意事项
- RTTI仅对具有虚函数的类(多态类型)有效。普通类型或非多态类的指针使用 dynamic_cast 或 typeid 将无法进行运行时判断。
- 性能开销:RTTI需要额外的类型信息存储和运行时检查,可能影响性能,尤其在嵌入式或高性能场景中应谨慎使用。
- 编译器支持:某些编译器允许禁用RTTI(如 GCC 的 -fno-rtti),此时 typeid 和 dynamic_cast 行为受限或不可用。
- typeid.name() 返回的名称是编译器相关的(通常为 mangled name),可使用 abi::__cxa_demangle 解析成可读形式(linux/GCC下)。
4. 实用技巧:打印可读类型名
使用 GCC/Clang 的扩展解析类型名:
#include <typeinfo>
#include <cxxabi.h>
#include <memory>
std::String demangle(const char* name) {
int status;
std::unique_ptr<char[], void(*)(void*)> demangled(
abi::__cxa_demangle(name, nullptr, nullptr, &status),
std::free
);
return demangled ? demangled.get() : name;
}
// 使用:
std::cout << demangle(typeid(*ptr).name()) << std::endl;
基本上就这些。RTTI 是 C++ 多态的重要补充,合理使用能提升代码灵活性和安全性。


