首先介绍一下为什么会引进多态呢,基于c++的复用性和拓展性而言,同类的程序模块进行大量重复,是一件无法容忍的事情,比如我设置了苹果,香蕉,西瓜类,现在想把这些东西都装到碗这个函数里,那么在主函数当中,声明对象是必须的,但是每一次装进碗里对于水果来说,都要用自己的指针调用一次装的功能,那为什么不把这些类抽象成一个水果类呢,直接定义一个水果类的指针一次性调用所有水果装的功能呢,这个就是利用父类指针去调用子类成员,但是这个思想受到了指针指向类型的限制,也就是说表面指针指向了子类成员,但实际上还是只能调用子类成员里的父类成员,这样的思想就变的毫无意义了,如果想要解决这个问题,只要在父类前加上virtual就可以解决了,这里就是利用虚函数实现多态的实例。

首先还是作为举例来两个类,在之前基础知识的帖子中提到过,空类的大小是一个字节(占位符),函数,静态变量都在编译期就形成了,不用类去分配空间,但是做一个小实验,看一看在定义了虚函数之后,类的大小是多少呢

#include <iostream>
using namespace std;
class A{
public:int i; //4Bytevirtual void func(){cout << __FUNCTION__ <<"() line: " << __LINE__<< endl;}virtual void func2(){cout << __FUNCTION__ <<"() line: " << __LINE__<< endl;}
};class B : public A{int j; //4Bytevoid func(){cout << __FUNCTION__ <<"() line: " << __LINE__<< endl;}
};int main(){A a;cout <<"虚函数表地址 = " <<(int *)(&a) << endl;cout << "sizeof(class A) = " << sizeof(A) << " " <<"sizeof(class B) =" << sizeof(B);//输出16,16return 0;
}

很明显类里装了一个 8个字节(64位)的东西,除了整形int,就是指针了,没错这里装的就是函数指针

先把这个代码,给抽象成图形进行理解,在这CFather为A,CSon为B

 此时就是一个单纯的继承的情况,不存在虚函数,然后我new一个对象,A *p = new A;那么 p -> AA(),必然是指向A类中的AA()函数,那么函数的调用有两种方式 一种函数名加()直接调用,一种是利用函数指针进行调用,在这里我想要调用子类的,就可以利用函数指针进行调用,假设出来两个函数指针,来指向B类中的两个成员函数,如果我父类想要调用子类成员,就可以通过 p指针去调用函数指针,再通过函数指针去调用成员函数

 每一个函数都可以用一个函数指针去指着,那么每一类中的函数指针都可以形成自己的一个表,这个就叫做虚函数表.

 那么在创建对象后,为什么类中会有8个字节的内存空间呢?

在C++的标准规格说明书中说到,编译器必需要保证虚函数表的指针存在于对象中最前面的位置(这是为了保证正确取到虚函数的偏移量)。这意味着我们通过对象实例的地址得到这张虚函数表,然后就可以遍历其中函数指针,并调用相应的函数。也就是说这四个字节的指针,代替了上图中(p->*pfn)()的作用,指向了函数指针,也就是说,在使用了虚函数的父类成员函数,虽然写的还是p->AA(),实际上却是,(p->*(vfptr[0])),而指向哪个虚函数表就由,创建的对象来决定

至此,就能理解如何用虚函数这个机制来实现多态的了

下面,我将分别说明“无覆盖”和“有覆盖”时的虚函数表的样子。没有覆盖父类的虚函数是毫无意义的。我之所以要讲述没有覆盖的情况,主要目的是为了给一个对比。在比较之下,我们可以更加清楚地知道其内部的具体实现。

无虚数覆盖

下面,再让我们来看看继承时的虚函数表是什么样的。假设有如下所示的一个继承关系:

请注意,在这个继承关系中,子类没有重载任何父类的函数。那么,在派生类的实例中,Derive d; 的虚函表:

我们可以看到下面几点:

1)虚函数按照其声明顺序放于表中。

2)父类的虚函数在子类的虚函数前面。

有虚数覆盖

覆盖父类的虚函数是很显然的事情,不然,虚函数就变得毫无意义。下面,我们来看一下,如果子类中有虚函数重载了父类的虚函数,会是一个什么样子?假设,我们有下面这样的一个继承关系。

为了让大家看到被继承过后的效果,在这个类的设计中,我只覆盖了父类的一个函数:f()。那么,对于派生类的实例,其虚函数表会是下面的一个样子:

我们从表中可以看到下面几点,

1)覆盖的f()函数被放到了虚表中原来父类虚函数的位置。

2)没有被覆盖的函数依旧。

这样,我们就可以看到对于下面这样的程序,

            Base *b = new Derive();

            b->f();

由b所指的内存中的虚函数表的f()的位置已经被Derive::f()函数地址所取代,于是在实际调用发生时,是Derive::f()被调用了。这就实现了多态。

*****************************************************************************************************************************************

现在将多继承的虚函数实现多态的情况讨论一下,另再加上从代码层面上对这个机制有更深的理解

讨论多继承还是从有无虚函数覆盖的情况来开始

无虚函数覆盖

假设有下面这样一个类的继承关系。注意:子类并没有覆盖父类的函数。

对于子类实例中的虚函数表,是下面这个样子:

我们可以看到:

1)  每个父类都有自己的虚表。

2)  子类的成员函数被放到了第一个父类的表中。(所谓的第一个父类是按照声明顺序来判断的)

这样做就是为了解决不同的父类类型的指针指向同一个子类实例,而能够调用到实际的函数。

有虚函数覆盖

下图中,我们在子类中覆盖了父类的f()函数。

下面是对于子类实例中的虚函数表的图:

我们可以看见,三个父类虚函数表中的f()的位置被替换成了子类的函数指针。这样,我们就可以任一静态类型的父类来指向子类,并调用子类的f()了。比如

Derive d;
Base1 *b1 = &d;
Base2 *b2 = &d;
Base3 *b3 = &d;b1->f(); //Derive::f()
b2->f(); //Derive::f()
b3->f(); //Derive::f()b1->g(); //Base1::g()
b2->g(); //Base2::g()
b3->g(); //Base3::g()

C++语言虚函数表实现多态原理相关推荐

  1. C++虚函数表实现详细解析 (附示例)

    目录 前言 运行环境 什么是虚函数 简单示例 虚函数的实现原理 虚函数表 通过虚表找到函数地址并调用 虚函数表的具体分析 无继承关系时的虚函数表 单继承但未重写虚函数时的虚函数表 单继承且重写虚函数时 ...

  2. 【C++】多态 —— 条件 | 虚函数重写 | 抽象类 | 多态的原理

    多态 1. 多态 2. 多态的定义和实现 2.1 多态的条件 2.2 虚函数重写的两个例外 2.2.1 协变 2.2.2 析构函数的重写 2.3 只有父类带 virtual 的情况 2.4 C++11 ...

  3. 线性表实现多项式相加c语言,用线性表实现多个多项式相加

    今天开始想复习一下数据结构,就从线性表开始吧. 今天是用线性表实现多个多项式相加这个题目,自变量是x. 题目描述如下: 在数学上,一个一元多项式Pn(x)可按降幂写成:Pn(x) = pn x^n + ...

  4. 【C++】多态(万字详解) —— 条件 | 虚函数重写 | 抽象类 | 多态的原理

  5. c语言虚函数是什么,C语言中什么函数不能声明为虚函数?

    2016-07-14 00:51齐智富 客户经理 --------------------------------------------------------------------------- ...

  6. c++远征之多态篇——虚函数及其实现原理

    以下内容源于慕课网的学习整理,如有侵权,请告知删除. 1.多态的定义 简单理解,就是对于同一条命令,不同对象会做出不同的操作. 相同对象收到不同消息,或者不同对象收到相同消息时,产生不同的动作. 2. ...

  7. C++虚函数的工作原理

    静态绑定与动态绑定 讨论静态绑定与动态绑定,首先需要理解的是绑定,何为绑定?函数调用与函数本身的关联,以及成员访问与变量内存地址间的关系,称为绑定. 理解了绑定后再理解静态与动态. 静态绑定:指在程序 ...

  8. C++虚函数的实现方式(虚表+虚指针)

    虚函数表实现原理 虚函数的实现是由两个部分组成的,虚函数指针与虚函数表. 用virtual关键字申明的函数叫做虚函数,虚函数肯定是类的成员函数. 存在虚函数的类都有一个一维的虚函数表叫做虚表.每一个类 ...

  9. 构造函数为什么不能是虚函数

    从存储空间角度看 虚函数相应一个指向vtable虚函数表的指针,这大家都知道,但是这个指向vtable的指针事实上是存储在对象的内存空间的. 问题出来了,假设构造函数是虚的.就须要通过 vtable来 ...

最新文章

  1. 远程教育中教师能力特点分析(转载)
  2. labview叠加白噪声_振荡器的相位噪声模型
  3. html制作固定列的表格,带固定列的简单HTML表格
  4. AIX errdemon 命令
  5. pytorch以特征图的输入方式训练LSTM模型
  6. 传感器 - 距离传感器
  7. mysql多表查询练习_MySQL多表查询综合练习答案
  8. 《Android游戏编程之从零开始》
  9. php网页可视化编辑器,推荐几款HTML可视化在线编辑器
  10. 虚拟机如何与主机之间直接复制粘贴文件(使用VMware Tools)
  11. 2.2 反相放大器、高输入电阻反相放大器、反相高压放大器
  12. 2017第48周日昨天休息
  13. 道可道,非常道 ---8个做事之“理”
  14. Python最假的库:Faker
  15. 华为机试—手机号码验证
  16. 中国环保乳胶漆市场供需调研及竞争策略分析报告2022-2028年
  17. 7-9 彩虹瓶 (25 分)(c++)
  18. SYD8811 GPIO模块中PAD和GPIO的关系
  19. /mnt 与 /mnt/的区别
  20. sap linux客户端,SAP 与 Linux类比

热门文章

  1. 【开发问题记录①】关于滑动CollectionView时ContentSize变化的问题
  2. 《延世大学韩国语教程2》第二十课 办公室(上)
  3. DockLayout布局
  4. Activity返回值
  5. 用EXCEL来解决同期比较的问题
  6. Vue CLI 3.0 正式发布,Vue.js 开发标准化工具
  7. python3.6安装pip3_python3.6如何安装pip
  8. 搭建SSM全流程框架过程
  9. mysql中表结构语句_mysql中表数据与表结构复制语句
  10. 安装mysql中文步骤_mysql安装步骤-Go语言中文社区