当前位置: 首页 > news >正文

dynamic_cast的理解

dynamic_cast:(具体使用就不详细说明了)

C++ 中用于 安全的类层次结构转换 的类型转换运算符,主要用于 多态类型(即包含虚函数的类)的指针或引用之间的转换

前提条件:

必须要有虚函数才能使用 dynamic_cast。即使存在单纯的继承关系,如果基类没有虚函数,dynamic_cast 无法使用

原因:

1. 为什么必须要有虚函数?

dynamic_cast 依赖 运行时类型信息(RTTI),而 RTTI 是通过虚函数表(vtable)实现的。

  • 虚函数表(vtable):当一个类包含虚函数时,编译器会为该类生成虚函数表,并在对象中插入指向该表的指针(vptr)。
  • RTTI 的存储:虚函数表中会保存类的类型信息(如类型名称、继承关系等),供 dynamic_cast 在运行时查询。
  • 没有虚函数 → 没有虚函数表 → 没有 RTTI → dynamic_cast 无法工作

2. 示例验证

代码示例(基类无虚函数)
#include <iostream>

class Base {};  // 基类没有虚函数
class Derived : public Base {};

int main() {
    Base* base = new Derived();
    
    // 尝试使用 dynamic_cast(编译错误!)
    Derived* derived = dynamic_cast<Derived*>(base);  
    
    delete base;
    return 0;
}
编译错误信息
error: 'Base' is not polymorphic
     Derived* derived = dynamic_cast<Derived*>(base);
                                     ^

3. 单纯继承无法使用 dynamic_cast

即使存在继承关系,只要基类没有虚函数:

  • dynamic_cast 无法编译通过,因为基类不是多态类型。
  • 无法进行安全的运行时类型检查

工作原理:

首先,dynamic_cast的核心在于运行时类型检查。这通常通过对象的虚函数表(vtable)中的RTTI指针来实现。每个多态类(即有虚函数的类)的虚函数表都包含一个指向type_info的指针,该指针提供了类的类型信息。当进行dynamic_cast时,编译器生成的代码会遍历继承树,检查目标类型是否是当前类型或其派生类(个人觉得是比较 hash_code(), 而不是name())

class type_info
{
public:

    type_info(const type_info&) = delete;
    type_info& operator=(const type_info&) = delete;

    size_t hash_code() const noexcept
    {
        return __std_type_info_hash(&_Data);
    }

    bool operator==(const type_info& _Other) const noexcept
    {
        return __std_type_info_compare(&_Data, &_Other._Data) == 0;
    }

#if !_HAS_CXX20
    bool operator!=(const type_info& _Other) const noexcept
    {
        return __std_type_info_compare(&_Data, &_Other._Data) != 0;
    }
#endif // !_HAS_CXX20

    bool before(const type_info& _Other) const noexcept
    {
        return __std_type_info_compare(&_Data, &_Other._Data) < 0;
    }

    const char* name() const noexcept
    {
        #ifdef _M_CEE_PURE
        return __std_type_info_name(&_Data, static_cast<__type_info_node*>(__type_info_root_node.ToPointer()));
        #else
        return __std_type_info_name(&_Data, &__type_info_root_node);
        #endif
    }

    const char* raw_name() const noexcept
    {
        return _Data._DecoratedName;
    }

    virtual ~type_info() noexcept;

private:

    mutable __std_type_info_data _Data;
};

性能开销来源:

  • 虚表访问: 需通过虚表获取 type_info,可能引发缓存未命中。
// 伪代码:获取动态类型
const __class_type_info* __get_dynamic_type(const void* obj) {
    // 获取虚表指针(通常位于对象起始位置)
    const void** vtable = *reinterpret_cast<const void***>(obj);
    // 虚表中存储了 type_info 指针(通常位于固定位置,如 -1 偏移)
    const __class_type_info* type_info = vtable[-1];
    return type_info;
}
  • 继承链遍历: 递归检查继承关系的复杂度为 O(继承深度)。
// 伪代码:检查继承关系
bool __do_find(const __class_type_info* target) const {
    if (this == target) {
        return true;  // 当前类型就是目标类型
    }

    // 遍历基类列表
    if (is_single_inheritance()) {
        return __base_type->__do_find(target);  // 单继承递归
    } else {
        for (each base in __base_info) {         // 多继承递归
            if (base.__base_type->__do_find(target)) {
                return true;
            }
        }
    }
    return false;
}
  • 指针调整计算: 多重继承中可能需要复杂偏移计算。(在多重继承中,不同基类的子对象可能有不同的偏移量,需调整指针:)
// 伪代码:调整指针偏移
void* adjust_pointer(const void* src, const __class_type_info* src_type, const __class_type_info* target_type) {
    // 计算从源类型到目标类型的偏移量
    ptrdiff_t offset = calculate_offset(src_type, target_type);
    return reinterpret_cast<char*>(src) + offset;
}

判断两个类是否有继承关系:

方法检查时机适用范围优点缺点
std::is_base_of编译时任意类之间的继承关系简单、标准不能检查多态对象的具体类型
typeiddynamic_cast运行时多态类型的具体对象可以检查实际对象类型依赖 RTTI,性能开销

推荐方法

  • 编译时检查:优先使用 std::is_base_of
  • 运行时检查:使用 typeiddynamic_cast(需多态类型)。

相关文章:

  • 使用OpenGL实现双线性插值和双三次插值C++实现
  • 【C++】类和对象(二)默认成员函数之拷贝构造函数、运算符重载
  • IPShocks:行星际激波数据库
  • 机器学习——Bagging、随机森林
  • leetcode day31 453+435
  • 代理服务器中的代理服务器与SSL协议有什么关系?
  • 元宇宙中的“数字护照“:代理IP如何重构虚拟世界的网络规则
  • 无序抓取系列(四)
  • linux的基础命令
  • 微信小程序登录和获取手机号
  • [学习笔记]NC工具安装及使用
  • 【AI News | 20250326】每日AI进展
  • 最大字段和问题 C++(穷举、分治法、动态规划)
  • h5运行在手机浏览器查看控制台信息
  • leetcode41.缺失的第一个正数
  • 数智读书笔记系列025《智能医疗:医学人工智能的未来》
  • Rust安装并配置配置vscode编译器
  • CPP从入门到入土之类和对象Ⅲ
  • UMI-OCR Docker 部署
  • Python:计算机二级:简单应用
  • 总导演揭秘十五运会闭幕式:赴一场星辰大海之约
  • 重庆大学通报本科生发14篇SCI论文处理结果
  • AI药企英矽智能第三次递表港交所:去年亏损超1700万美元,收入多数来自对外授权
  • 安徽亳州涡阳县司法局党组书记刘兴连落马
  • 陕西永寿4岁女童被蜜蜂蜇伤致死,当地镇政府介入处理
  • 【社论】以法治力量促进民企长远健康发展