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

调用基类的纯虚函数,如何知道纯虚函数会调用哪个派生类(子类)中的实现。

在 C++ 中,调用基类的纯虚函数实际上是通过运行时多态性来决定调用哪一个派生类的实现。这种机制是通过虚函数表(vtable)和虚函数指针(vptr)实现的。下面我们来详细探讨一下这个过程。

虚函数表和虚函数指针

  1. 虚函数表(vtable):

    • 每个包含虚函数的类(包括纯虚函数)都会有一个虚函数表。虚函数表是一个指针数组,每个指针指向类的虚函数的具体实现。
    • 虚函数表是编译器在编译时生成的,并且对于同一个类的所有对象是共享的。
  2. 虚函数指针(vptr):

    • 每个对象有一个指向其类的虚函数表的指针,称为虚函数指针(vptr)。
    • 当一个对象被创建时,其 vptr 被初始化为指向该对象所属类的虚函数表。

当调用一个虚函数时,程序会通过对象的 vptr 找到相应的 vtable,并在 vtable 中找到该函数的地址,然后进行调用。这种机制允许程序在运行时根据对象的实际类型调用适当的函数实现,这就是多态性。

调用纯虚函数的过程

假设你有一个基类 Base 和几个派生类 Derived1Derived2,基类 Base 定义了一个纯虚函数 doSomething。以下是如何知道调用哪个派生类实现的步骤:

对于 obj2->doSomething(),类似的过程会发生,但它的 vptr 指向 Derived2 的 vtable,最终调用 Derived2doSomething 的实现。

运行时确定派生类的实现

这是因为 C++ 的多态性允许基类指针(或引用)指向派生类对象。调用虚函数时,实际调用的函数实现是通过对象的动态类型(即它真正的派生类类型)来确定的。这种类型是在运行时决定的,而不是编译时。

代码示例

下面是一个完整的代码示例,展示了上述过程:

  1. 定义类和函数:

    class Base {
    public:virtual void doSomething() = 0; // 纯虚函数
    };class Derived1 : public Base {
    public:void doSomething() override {std::cout << "Derived1 implementation" << std::endl;}
    };class Derived2 : public Base {
    public:void doSomething() override {std::cout << "Derived2 implementation" << std::endl;}
    };
    
  2. 实例化派生类对象:

    Base* obj1 = new Derived1();
    Base* obj2 = new Derived2();
    
  3. 调用虚函数:

    obj1->doSomething(); // 调用 Derived1 的实现
    obj2->doSomething(); // 调用 Derived2 的实现
    

 

决定调用哪个派生类实现的过程

当你调用 obj1->doSomething() 时,以下过程发生:

  1. 查找 vptr:

    • obj1 是指向 Derived1 对象的基类指针。
    • 程序通过 obj1 找到它的 vptr,该 vptr 指向 Derived1 的 vtable。
  2. 查找 vtable:

    • 程序查找 Derived1 的 vtable,这个表包含 doSomething 的地址。
  3. 调用函数:

    • 程序通过 vtable 获取 doSomething 的地址,然后调用这个地址处的函数,即 Derived1doSomething 的实现。

对于 obj2->doSomething(),类似的过程会发生,但它的 vptr 指向 Derived2 的 vtable,最终调用 Derived2doSomething 的实现。

运行时确定派生类的实现

这是因为 C++ 的多态性允许基类指针(或引用)指向派生类对象。调用虚函数时,实际调用的函数实现是通过对象的动态类型(即它真正的派生类类型)来确定的。这种类型是在运行时决定的,而不是编译时。

代码示例

下面是一个完整的代码示例,展示了上述过程:

#include <iostream>class Base {
public:virtual void doSomething() = 0; // 纯虚函数
};class Derived1 : public Base {
public:void doSomething() override {std::cout << "Derived1 implementation" << std::endl;}
};class Derived2 : public Base {
public:void doSomething() override {std::cout << "Derived2 implementation" << std::endl;}
};int main() {Base* obj1 = new Derived1();Base* obj2 = new Derived2();obj1->doSomething(); // 输出: Derived1 implementationobj2->doSomething(); // 输出: Derived2 implementationdelete obj1;delete obj2;return 0;
}

在这个示例中,通过基类指针调用 doSomething 时,程序根据实际的派生类类型调用相应的实现,这展示了 C++ 中的运行时多态性。

通过调试查看

如果你使用调试器(如 gdb),你可以在调用虚函数前设置断点,并逐步查看调用过程。你会看到程序通过 vptr 查找 vtable,然后调用适当的函数实现。这是验证多态行为的一个好方法。

总结

  • vptr 和 vtable: vptr 指向对象的 vtable,通过它们在运行时决定调用哪个派生类的实现。
  • 多态性: 基类指针或引用调用虚函数时,实际调用的是派生类的实现,这通过动态绑定实现。
  • 调试和分析: 使用调试器可以更深入地观察这种运行时行为。
http://www.lryc.cn/news/388395.html

相关文章:

  • 塑造卓越企业家IP:多维度视角下的策略解析
  • Rust 跨平台-Android 和鸿蒙 OS
  • Typora导出为Word
  • 数据库被后台爆破如何解决?
  • php7.4源码安装dbase7.1.1扩展
  • OkHttp的源码解读1
  • 08:结构体
  • 喜讯!安全狗荣获“2023年网络安全技术支撑优秀单位”称号
  • android里面json操作
  • MATLAB的.m文件与Python的.py文件:比较与互参
  • 武汉星起航:自运营团队精准把握亚马逊红利,引领跨境电商新潮流
  • 嵌入式计算器模块实现
  • tomcat定时重启
  • 构建LangChain应用程序的示例代码:48、如何使用非文本生成工具创建多模态代理
  • 【笔记】记录一次全新的Java项目部署过程
  • 达梦数据库系列—14. 表空间的备份和还原
  • 奔驰G350升级原厂自适应悬挂系统有哪些作用
  • 一个启动脚本例子
  • grpc学习golang版( 六、服务器流式传输 )
  • ubuntu语音库ALSA报错具体原因
  • Java高级重点知识点-17-异常
  • DM达梦数据库函数分析(与mysql对应函数区别及用法分析)
  • ROS2用c++开发参数节点通信
  • docker 部署jitsi meet
  • 【Pytest自动化测试详解】
  • 6-14题连接 - 高频 SQL 50 题基础版
  • 深度挖掘数据资产,洞察业务先机:利用先进的数据分析技术,精准把握市场趋势,洞悉客户需求,为业务决策提供有力支持,实现持续增长与创新
  • 亚马逊广告如何设置关键词竞价获取最优广告投入产出比 (ACOS)
  • vision mamba-yolov8:结合Vmamba的yolov8目标检测改进实现
  • 2025秋招NLP算法面试真题(十一)-Transformer的并行化