C++中虚表是什么意思(概念及示例)

 更新时间:2024年03月19日 08:55:45   作者:江水为竭  
虚函数表,以及虚函数指针是实现多态性(Polymorphism)的关键机制,这篇文章主要介绍了C++中虚表是什么意思(概念及示例),需要的朋友可以参考下

虚函数表,以及虚函数指针是实现多态性(Polymorphism)的关键机制。多态性允许我们通过基类的指针或引用来调用派生类的函数

定义

虚函数(Virtual Function)

  • 定义:类中使用virtual 关键字修饰的函数 叫做虚函数

  • 语法

highlighter- cpp

class Base {
public:
    virtual void show() { cout << "Base show" << endl; }
};

虚函数表(Virtual Function Table)

  • 定义:当类含有至少一个虚函数时,编译器会为该类创建一个虚函数表。这个表是一个编译时构建的静态数组,存储了指向类中所有虚函数的指针。如果一个派生类重写了这些函数,那么在派生类的虚表中,相应函数的指针会被更新为指向派生类中的版本。
  • 作用v-table使得在运行时可以实现函数的动态绑定,允许通过基类的指针或引用调用正确的函数版本。

虚函数指针(Virtual Pointer)

  • 定义:每个含有虚函数的类的对象(实例化出的)会持有一个指向相应虚表的指针,这个指针通常被称为虚指针(vptr)。vptr是对象运行时的一部分,确保了当通过基类指针调用虚函数时,能够查找到正确的函数实现。
  • 作用:在对象的生命周期开始时,构造函数会设置vptr以指向相应的虚函数表。如果有派生类对象,它的构造函数会更新vptr,以指向派生类的虚函数表。这保证了通过基类的引用或指针调用虚函数也会执行最派生类的重写版本。

示例

highlighter- cpp

#include <iostream>
using namespace std;
class Base {
public:
    virtual void func1() { cout << "Base::func1" << endl; }
    virtual void func2() { cout << "Base::func2" << endl; }
};
class Derived : public Base {
public:
    void func1() override { cout << "Derived::func1" << endl; }
    // func2() 继承自 Base
};
void printVTable(void* obj) {
    cout << "vptr Address: " << obj << endl;
    void** vTable = *(void***)obj;
    cout << "VTable[0] (func1): " << vTable[0] << endl;
    cout << "VTable[1] (func2): " << vTable[1] << endl;
}
int main() {
    Base* base = new Base();
    Derived* derived = new Derived();
    cout << "Base object:" << endl;
    printVTable(base);
    cout << "\nDerived object:" << endl;
    printVTable(derived);
    delete base;
    delete derived;
    return 0;
}

程序输出如下,可以看到没用重写的func2函数地址是一样的。

highlighter- apache

Base object:
vptr Address: 0x8c1510   
VTable[0] (func1): 0x422270
VTable[1] (func2): 0x4222b0
Derived object:
vptr Address: 0x8c1530   
VTable[0] (func1): 0x422330
VTable[1] (func2): 0x4222b0

如下图所示:

面试题

(来自2024腾讯实习面试)场景题:一个类 A,里面有一个打印 helloworld 的虚函数,然后类 A 会在构造函数里调用这个虚函数,此时有个类 B,继承A,重写了这个 helloworld虚函数,问你在创建类 B 时,会打印 A 里的 helloworld 还是 B 里的。

代码如下:

highlighter- arduino

class A {
public:
    A() {
        print();
    }
    virtual void print() {
        cout << "A print" << endl;
    }
};
class B : public A {
public:
    void print() override {
        cout << "B print" << endl;
    }
};
int main() {
    A* aTemp = new B();
    delete aTemp;
}

解答:基类构造函数执行的时候,派生类的部分尚未初始化,因此调用的虚函数不会下发到派生类中。

最终会打印 A print,而不是类 B 里重写的版本

到此这篇关于C++中虚表是什么的文章就介绍到这了,更多相关C++虚表内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++实现LeetCode(2.两个数字相加)

    C++实现LeetCode(2.两个数字相加)

    这篇文章主要介绍了C++实现LeetCode(两个数字相加),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++实现LeetCode(22.生成括号)

    C++实现LeetCode(22.生成括号)

    这篇文章主要介绍了C++实现LeetCode(22.生成括号),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++设计模式之状态模式

    C++设计模式之状态模式

    这篇文章主要介绍了C++设计模式之状态模式,本文讲解了什么是状态模式、状态模式的使用场合、状态模式的实现代码等内容,需要的朋友可以参考下
    2014-10-10
  • 深入理解C/C++混合编程

    深入理解C/C++混合编程

    本篇文章是对C/C++混合编程进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • 简要介绍C++编程中的友元函数和友元类

    简要介绍C++编程中的友元函数和友元类

    这篇文章主要介绍了C++编程中的友元函数和友元类,属于较为冷僻的知识,在实际开发中较少使用,需要的朋友可以参考下
    2015-09-09
  • Linux管道揭秘之匿名管道连接进程世界的方法

    Linux管道揭秘之匿名管道连接进程世界的方法

    文章介绍了Linux中的管道(Pipe)概念,包括其定义、作用、类型、工作原理以及如何在父子进程间使用,匿名管道是进程间通信的一种机制,通过pipe()系统调用创建,具有读端和写端文件描述符,文章详细解释了匿名管道的创建、使用流程、4种情况和5种特性
    2024-11-11
  • C++示例讲解string容器

    C++示例讲解string容器

    c++相比c的一个好处就是实现了很多的容器和泛型算法,使得程序员的工作得到了很大的简化,本文重点给大家介绍C++string容器基本概念讲解,需要的朋友参考下吧
    2022-07-07
  • C++基于字符串实现大数相乘问题的代码详解

    C++基于字符串实现大数相乘问题的代码详解

    在实际编程中,我们经常会遇到需要处理大整数的情况,由于编程语言中内置整数类型有其表示范围的限制,当需要处理的整数超出这些范围时,就不能直接使用内置类型进行计算,所以本文给大家介绍了相关的解决方法,需要的朋友可以参考下
    2025-03-03
  • 更优雅的C++字符串格式化实现方法详解

    更优雅的C++字符串格式化实现方法详解

    在用C++编写代码时,经常需要用到字符串拼接及格式化,尤其是在拼写sql语句时。所以本文为大家介绍了更优雅的C++字符串格式化实现方法,希望对大家有所帮助
    2023-04-04
  • C++ OpenCV实战之文档照片转换成扫描文件

    C++ OpenCV实战之文档照片转换成扫描文件

    这篇文章主要为大家介绍一个C++ OpenCV的实战——文档照片转换成扫描文件,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2022-09-09

最新评论