C++新特性 强制转换 dynamic_cast

dynamic_cast

用于具有虚函数的基类派生类之间的指针或引用

  • 基类必须具备虚函数
    • 原因:dynamic_cast 是运行时类型检查,需要运行时类型信息(RTTI),而这个信息是存储与类的虚函数表关系紧密,只有一个类定义了虚函数,才会有虚函数表
  • 运行时检查,转型不成功则返回一个空指针
  • 非必要不要使用 dynamic_cast, 有额外的函数开销

常见的转换方式:

  • 基类指针或引用转派生类指针(必须使用dynamic_cast)
  • 派生类指针或引用转基类指针(可以使用 dynamic_cast, 但是更推荐使用 static_cast)
#include <iostream>
using namespace std;

// 基类与派生类之间的转换

class CFather {
public:
    CFather() {

    }

    virtual void foo() {
        std::cout << "CFather()::void foo()" << std::endl;
    }

    int m_nTest;
};

class CSon : public CFather {
public:
    virtual void foo() {
        std::cout << "CSon::void foo()" << std::endl;
    }

    int m_nSon;
};

int main() {
    CFather f;
    CSon s;

    CFather* pFather = &f;
    CSon* pSon = &s;

    // 使用 dynamic 前提

    // pFather = static_cast<CFather*>(pSon);

    // 向下转换 父类转子类 不安全

    // 有一种语法能检测出这种转换是不安全的, dynamic_cast
    // 在运行时刻检测转换是否安全
    // pSon = static_cast<CSon*>(pFather);
    // pSon->m_nSon = 123;

    // dynamic_cast 能够在运行的时刻,检测出被转换的指针的类型(依赖RTTI)
    // 有额外的开销,一般而言只有在向下转换时才必须使用
    pSon = dynamic_cast<CSon*>(pFather);

    if (pSon != nullptr) {
        pSon->m_nSon = 123;
        std::cout << pSon->m_nSon << std::endl;
    }

    if (pSon == nullptr) std::cout << "null" << std::endl;


    // 向上转换 子类转父类 安全
    pFather = static_cast<CFather*>(pSon);

    // 具有多态类类类型的向下转换时必须使用,其余情况可以不用
}
posted @ 2022-08-05 14:37  岁月飞扬  阅读(37)  评论(0编辑  收藏  举报