C++中typeid如何实现RTTI运行时类型识别?

来源:Vuejs社区作者:孙悟空头衔:草根站长
导读:本期聚焦于小伙伴创作的《C++中typeid如何实现RTTI运行时类型识别?》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《C++中typeid如何实现RTTI运行时类型识别?》有用,将其分享出去将是对创作者最好的鼓励。

C++的RTTI(运行时类型识别)机制允许程序在运行阶段获取对象的真实类型信息,typeid是RTTI的核心操作符之一,通过它可以查询表达式的类型,在多态场景下的类型判断中应用非常广泛。

C++中typeid如何实现RTTI运行时类型识别?

typeid的基本用法

typeid的操作数可以是类型名称,也可以是表达式,它的返回结果是std::type_info类型的常引用。使用typeid需要先包含<typeinfo>头文件,以下是基础的使用示例:

#include <iostream>
#include <typeinfo>

int main() {
    int num = 10;
    double val = 3.14;
    // 操作类型为int
    std::cout << "int类型的type_info名称: " << typeid(int).name() << std::endl;
    // 操作表达式为int变量
    std::cout << "num变量的类型名称: " << typeid(num).name() << std::endl;
    // 操作表达式为double变量
    std::cout << "val变量的类型名称: " << typeid(val).name() << std::endl;
    return 0;
}

typeid处理多态类型的特点

当typeid的操作数是带有虚函数的类的指针或引用时,它会在运行时确定指针或引用指向的实际对象类型,而不是指针或引用本身的静态类型。如果操作数不是多态类型(没有虚函数),则typeid在编译阶段就会确定类型。

下面通过示例说明多态场景下的typeid表现:

#include <iostream>
#include <typeinfo>

// 基类,包含虚函数,属于多态类型
class Base {
public:
    virtual ~Base() {}
};

// 派生类1
class DerivedA : public Base {};

// 派生类2
class DerivedB : public Base {};

int main() {
    Base* ptr1 = new DerivedA();
    Base* ptr2 = new DerivedB();
    Base obj;

    // ptr1指向DerivedA对象,运行时识别为DerivedA
    std::cout << "ptr1指向的对象类型: " << typeid(*ptr1).name() << std::endl;
    // ptr2指向DerivedB对象,运行时识别为DerivedB
    std::cout << "ptr2指向的对象类型: " << typeid(*ptr2).name() << std::endl;
    // 基类对象,静态类型为Base
    std::cout << "obj的类型: " << typeid(obj).name() << std::endl;

    delete ptr1;
    delete ptr2;
    return 0;
}

std::type_info的常用属性

std::type_info类没有提供可移植的构造函数,通常只能通过typeid获取它的实例,它有两个常用成员:

  • name():返回类型的名称字符串,不同编译器的返回结果可能有差异,比如GCC会返回修饰后的类型名,MSVC会返回可读的类型名。
  • operator==operator!=:用于比较两个type_info对象是否对应同一个类型,这是判断类型一致性的可靠方式,不要直接比较name()的返回值。

类型判断的示例如下:

#include <iostream>
#include <typeinfo>

class Base {
public:
    virtual ~Base() {}
};

class Derived : public Base {};

int main() {
    Base* basePtr = new Derived();
    // 判断实际类型是否为Derived
    if (typeid(*basePtr) == typeid(Derived)) {
        std::cout << "basePtr指向的是Derived类型对象" << std::endl;
    } else {
        std::cout << "basePtr指向的不是Derived类型对象" << std::endl;
    }
    // 判断静态类型是否一致
    Base obj;
    if (typeid(obj) != typeid(Derived)) {
        std::cout << "obj的静态类型是Base,不是Derived" << std::endl;
    }
    delete basePtr;
    return 0;
}

使用typeid的注意事项

在使用typeid进行RTTI识别时,需要注意以下几点:

  • 不要对空指针解引用后使用typeid,比如typeid(*nullptr)会抛出std::bad_typeid异常。
  • typeid只能识别有完整定义的类型,如果操作数是不完整类型,会导致编译错误。
  • RTTI会带来一定的性能开销,因为需要在运行时查询类型信息,在对性能要求极高的场景需要谨慎使用。
  • 如果类没有虚函数,即使通过基类指针指向派生类对象,typeid也无法识别派生类类型,只能得到静态类型。

typeid与dynamic_cast的配合

在实际开发中,typeid经常和dynamic_cast配合使用,先通过typeid判断对象类型,再使用dynamic_cast进行安全的类型转换,避免转换失败的问题。

#include <iostream>
#include <typeinfo>

class Base {
public:
    virtual ~Base() {}
    virtual void func() {}
};

class Derived : public Base {
public:
    void derivedFunc() {
        std::cout << "调用Derived类的特有方法" << std::endl;
    }
};

int main() {
    Base* ptr = new Derived();
    // 先判断类型
    if (typeid(*ptr) == typeid(Derived)) {
        // 安全转换
        Derived* derivedPtr = dynamic_cast<Derived*>(ptr);
        if (derivedPtr) {
            derivedPtr->derivedFunc();
        }
    }
    delete ptr;
    return 0;
}

C++typeidRTTI运行时类型识别修改时间:2026-06-21 04:18:36

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。