type_info 不能用于类型识别,因其仅提供只读名称和哈希码,无标准格式、不可构造拷贝、hash_code不跨模块稳定、name()字符串实现依赖且无法安全匹配;正确方式是 typeid 配合 == 或 dynamic_cast,二者均需开启 rtti。

不能直接用 type_info 做类型比较或转换,它只提供只读的运行时类型名称和基础标识能力;真要识别/判断类型,得靠 dynamic_cast 或 typeid 配合 ==,且必须开启 RTTI。
为什么 type_info 本身不能用来“识别类型”
type_info 是个不可构造、不可拷贝的类,设计上就是只读句柄——它不暴露类型 ID 数值,也不支持哈希或比较以外的操作。你拿到一个 type_info&,只能调用 name()(返回实现定义的字符串)或 hash_code()(C++11 起),但 hash_code() 在不同编译单元可能不一致,不能跨 DLL/so 比较。
-
typeid(obj)返回的是const type_info&,不是新对象,多次调用同一对象得到的引用可能相等,但标准不保证地址相同 -
name()返回的字符串没有标准化格式,GCC 输出"St6vectorIiSaIiEE",MSVC 是class std::vector<int std::allocator> >"</int>,没法安全字符串匹配 - 试图
static_cast<const type_info>(&typeid(T))</const>是未定义行为,type_info没有公开的虚表布局
typeid 和 dynamic_cast 才是识别类型的正确组合
真正做类型识别(比如判断指针是否指向某个派生类),靠的是 typeid 的相等比较或 dynamic_cast 的空检查,二者都依赖 RTTI 且语义不同:
十天学会易语言图解教程用图解的方式对易语言的使用方法和操作技巧作了生动、系统的讲解。需要的朋友们可以下载看看吧!全书分十章,分十天讲完。 第一章是介绍易语言的安装,以及运行后的界面。同时介绍一个非常简单的小程序,以帮助用户入门学习。最后介绍编程的输入方法,以及一些初学者会遇到的常见问题。第二章将接触一些具体的问题,如怎样编写一个1+2等于几的程序,并了解变量的概念,变量的有效范围,数据类型等知识。其后,您将跟着本书,编写一个自己的MP3播放器,认识窗口、按钮、编辑框三个常用组件。以认识命令及事件子程序。第
- 用
typeid:适合已知对象存在(非空)、且需要精确类型匹配的场景,比如日志、断言、简单工厂分发if (typeid(*ptr) == typeid(DerivedA)) { /* 确实是 DerivedA,不是子类 */ } - 用
dynamic_cast:适合多态指针/引用的“安全向下转型”,能处理继承链,返回nullptr或抛出异常(引用版),比typeid更健壮if (auto* d = dynamic_cast<DerivedB*>(ptr)) { /* ptr 实际指向 DerivedB 或其派生类 */ } - 两者都不能用于非多态类型(即不含虚函数的类),否则
typeid仍工作但只反映静态类型,dynamic_cast编译失败
RTTI 开关和常见编译错误
RTTI 默认开启,但很多嵌入式或性能敏感项目会关掉(如 GCC/Clang 的 -fno-rtti,MSVC 的 /GR-)。关掉后:typeid 对非内置类型行为未定义,dynamic_cast 直接报错,而 type_info 类型本身仍存在,但无法实例化。
立即学习“C++免费学习笔记(深入)”;
- 错误信息类似:
error: cannot use 'typeid' with '-fno-rtti'或error C2680: 'dynamic_cast' requires the '/GR' option - 若链接时出现
undefined reference to 'typeinfo for XXX',说明某处用了typeid或异常,但目标文件没开 RTTI,需统一编译选项 - 启用 RTTI 不显著影响性能,除非高频调用
typeid或dynamic_cast;现代编译器对typeid常量比较能内联优化
真正麻烦的是跨模块类型识别:DLL/so 中的类,如果导出符号不一致或 RTTI 表未导出(如 Windows 上没加 __declspec(dllexport)),typeid 比较会失败,dynamic_cast 可能返回空——这时候别硬刚 type_info,该用接口抽象就用接口抽象。








