虚函数表-C++多态的实现原理解析

 更新时间:2021年02月01日 08:41:45   作者:sherlock_lin  
这篇文章主要介绍了虚函数表-C++多态的实现原理,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

参考:http://c.biancheng.net/view/267.html

1、说明

我们都知道多态指的是父类的指针在运行中指向子类,那么它的实现原理是什么呢?答案是虚函数表

关于virtual 一文中,我们详细了解了C++多态的使用方式,我们知道没有 virtual 关键子就没法使用多态

2、虚函数表

我们看一下下面的代码

class A
{
public:
  int i;
  virtual void func() { cout << "A func" << endl; }
  virtual void func2() { cout << "A func2" << endl; }
  void func3() { cout << "A func3" << endl; }
};
class B : public A
{
  int j;
  void func() { cout << "B func" << endl; }
  void func3() { cout << "B func3" << endl; }
};
int main()
{
  cout << sizeof(A) << ", " << sizeof(B); //输出 8,12
  return 0;
}

在32位编译模式下,程序的运行结果是:8,12

但是如果把代码中的 virtual 删掉,则程序的运行结果为:4,8

可以发现,有了虚函数之后,类所占的存储空间比没有虚函数多了4个字节,这个4个字节就是实现多态的关键 -- 位于对象存储空间的最前端的指针,存放的是 虚函数表的地址,这个是由编译器实现的

每个带有虚函数的类(包括其子类)都有虚函数表

虚函数表中存放着虚函数的地址,注意是虚函数的地址,非虚函数不在此列

虚函数表是编译器实现的,程序运行时被载入内存,一个类的虚函数表中列出了该类的全部虚函数地址。

例如,上面代码中,类A的对象的存储空间以及虚函数表如图所示:

类B的对象的存储空间以及虚函数表,如下图所示:

多态的函数调用语句被编译成根据基类指针所指向的对象中存放的虚函数表的地址,在虚函数表中查找虚函数地址,并调用虚函数的一系列指令

3、代码示例

在上面代码的基础上

A* p = new B();
p->func();	//B func
p->func3();	//A func3
p->func2();	//A func

第二行代码执行如下:

  • 取出 p 指针所指向的位置的前4个字节,即对象所属的类(类B)的虚函数表的地址(64位编译模式下是8个字节);
  • 根据虚函数表的地址找到虚函数表,并在虚函数表中查找要调用的虚函数地址;
  • 调用虚函数;

到此,我们应该不难理解,上面第二行和第三行代码执行的分别是类A和类B的方法

执行 p->func(); 找的是类B虚函数表中 func() 地址,因为类B重写了,所以保存的是类B的func()地址

而执行 p->func3(); 的时候,发现 func3() 不是虚函数,所以并没有找虚函数列表,而是直接调用的p(类A类型)的方法

同样的,执行 p->func2(); 的时候,找的也是类B的虚函数表,因为类B没有重写 func2,所以存的是类A的虚函数 func2() 的地址,所以执行了类A的 func2() 方法

到此这篇关于虚函数表-C++多态的实现原理的文章就介绍到这了,更多相关C++虚函数表实现多态原理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言实现变色进度条

    C语言实现变色进度条

    这篇文章主要为大家详细介绍了C语言实现一个变色的进度条,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • C++ 解引用与函数基础详解之内存地址、调用方法及声明

    C++ 解引用与函数基础详解之内存地址、调用方法及声明

    函数是C++ 中重要的编程概念,它们可以提高代码的可重用性、可读性和可维护性,本文介绍C++ 解引用与函数基础详解之内存地址、调用方法及声明,感兴趣的朋友跟随小编一起看看吧
    2024-04-04
  • C++实现LeetCode(48.旋转图像)

    C++实现LeetCode(48.旋转图像)

    这篇文章主要介绍了C++实现LeetCode(48.旋转图像),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • c语言和c++语言中const修饰的变量区别浅析

    c语言和c++语言中const修饰的变量区别浅析

    这篇文章主要给大家介绍了关于c语言和c++语言中const修饰的变量区别的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2022-02-02
  • 如何判断一个数是否为2的幂次方?若是,并判断出来是多少次方?

    如何判断一个数是否为2的幂次方?若是,并判断出来是多少次方?

    本篇文章是对如何判断一个数是否为2的幂次方?若是,并判断出来是多少次方的实现方法,进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • MATLAB中count函数用法示例代码

    MATLAB中count函数用法示例代码

    这篇文章主要介绍了MATLAB中的count函数,用于计算字符串中模式的出现次数,它可以处理单个模式和模式数组,并且可以忽略大小写,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2025-02-02
  • QT实现FTP上传文件

    QT实现FTP上传文件

    这篇文章主要为大家详细介绍了QT实现FTP上传文件,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • c++ 调用python传输图片实例

    c++ 调用python传输图片实例

    今天小编就为大家分享一篇c++ 调用python传输图片实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-12-12
  • c语言中十六进制转二进制显示的实现方法

    c语言中十六进制转二进制显示的实现方法

    本篇文章对c语言中十六进制转二进制显示的实现方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • 深入理解C语言sizeof()计算空间大小为8的问题

    深入理解C语言sizeof()计算空间大小为8的问题

    本文将介绍C语言中的sizeof()函数,以及如何使用它来计算变量、数据类型和数组在内存中的大小,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09

最新评论