虚函数、虚指针和虚表

http://eriol.iteye.com/blog/1167737

关于虚函数的背景知识

  1. 用virtual关键字申明的函数叫做虚函数,虚函数肯定是类的成员函数。
  2. 存在虚函数的类都有一个一维的虚函数表叫做虚表。类的对象有一个指向虚表开始的虚指针。虚表是和类对应的,虚表指针是和对象对应的。
  3. 多态性是一个接口多种实现,是面向对象的核心。分为类的多态性和函数的多态性。
  4. 多态用虚函数来实现,结合动态绑定。
  5. 纯虚函数是虚函数再加上= 0。并且该函数只有声明,没有实现。
  6. 抽象类是指包括至少一个纯虚函数的类。

那虚函数是如何运行的呢?

Cpp代码  
  1. class Base
  2. {
  3. public:
  4. virtual void func() {}
  5. }
  6. class Derive : public Base
  7. {
  8. public:
  9. void func() {}
  10. }
  11. void main()
  12. {
  13. Derive d;
  14. Base *pb = &d;
  15. b->func();
  16. }

编译器在编译的时候,发现Base类中有虚函数,此时编译器会为每个包含虚函数的类创建一个虚表(即vtable),该表是一个一维数组,在这个数组中存放每个虚函数的地址。由于Base类和Derive类都包含了一个虚函数func(),编译器会为这两个类都建立一个虚表,(即使子类里面没有virtual函数,但是其父类里面有,所以子类中也有了)

那么如何定位虚表呢?编译器另外还为每个类的对象提供了一个虚表指针(即vptr),这个指针指向了对象所属类的虚表。在程序运行时,根据对象的类型去初始化vptr,从而让vptr正确的指向所属类的虚表。所以在调用虚函数时,就能够找到正确的函数。

对于上述程序,由于pb实际指向的对象类型是Derive,因此vptr指向的Derive类的vtable,当调用pb->func()时,根据虚表中的函数地址找到的就是Derive类的func()函数。

正是由于每个对象调用的虚函数都是通过虚表指针来索引的,也就决定了虚表指针的正确初始化是非常重要的。换句话说,在虚表指针没有正确初始化之前,我们不能够去调用虚函数。那么虚表指针在什么时候,或者说在什么地方初始化呢?

答案是在构造函数中进行虚表的创建和虚表指针的初始化

还记得构造函数的调用顺序吗,在构造子类对象时,要先调用父类的构造函数,此时编译器只“看到了”父类,并不知道后面是否后还有继承者,它初始化父类对象的虚表指针,该虚表指针指向父类的虚表。当执行子类的构造函数时,子类对象的虚表指针被初始化,指向自身的虚表。对于以上的例子,当Derive类的d对象构造完毕后,其内部的虚表指针也就被初始化为指向Derive类的虚表。在类型转换后,调用pb->func(),由于pb实际指向的是Derive类的对象,该对象内部的虚表指针指向的是Derive类的虚表,因此最终调用的是Derive类的func()函数。

要注意:对于虚函数调用来说,每一个对象内部都有一个虚表指针,该虚表指针被初始化为本类的虚表。所以在程序中,不管你的对象类型如何转换,但该对象内部的虚表指针是固定的,所以呢,才能实现动态的对象函数调用,这就是C++多态性实现的原理。

总结(基类有虚函数):

  1. 每一个类都有虚表。
  2. 虚表可以继承,如果子类没有重写虚函数,那么子类虚表中仍然会有该函数的地址,只不过这个地址指向的是基类的虚函数实现。如果基类有3个虚函数,那么基类的虚表中就有三项(虚函数地址),派生类也会有虚表,至少有三项,如果重写了相应的虚函数,那么虚表中的地址就会改变,指向自身的虚函数实现。如果派生类有自己的虚函数,那么虚表中就会添加该项。
  3. 派生类的虚表中虚函数地址的排列顺序和基类的虚表中虚函数地址排列顺序相同。

虚函数、虚指针和虚表相关推荐

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

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

  2. 38.【C++ 虚函数 纯虚函数 虚基类 (最全详解)】

    虚函数.虚基类.纯虚函数 (一).虚函数 1.什么是虚函数: 2.虚函数的格式: 3.关于虚函数的注意事项: 4.虚函数的作用: 5.虚函数访问格式 6.虚函数的各种疑难杂症 [当指针是基类.但虚函数 ...

  3. 虚函数 虚继承 抽象类

    虚函数.纯虚函数.虚基类.抽象类.虚函数继承.虚继承 虚函数:虚函数是C++中用于实现多态(polymorphism)的机制.核心理念就是通过基类访问派生类定义的函数.是C++中多态性的一个重要体现, ...

  4. 图解C++虚函数 虚函数表

    图解C++虚函数 2016年07月02日 17:47:17 海枫 阅读数:5181 标签: 虚函数c++g++对象模型C++虚函数更多 个人分类: C/C++/linux 版权声明:本文为博主原创文章 ...

  5. 虚函数 纯虚函数 虚基类说明

    原文:http://www.cnblogs.com/ms-frank/archive/2008/01/16/1041310.html 虚基类 在说明其作用前先看一段代码 [cpp] view plai ...

  6. c语言中虚函数和纯虚函数,虚函数和纯虚函数的区别是什么?

    虚函数和纯虚函数的区别:1.纯虚函数只有定义,没有实现:而虚函数既有定义,也有实现的代码.2.包含纯虚函数的类不能定义其对象,而包含虚函数的则可以. 相关推荐:<C++视频教程> 虚函数( ...

  7. 【C++】多态 - 虚函数/虚析构函数以及虚函数表

    什么是多态: 指不同对象收到相同消息时或相同对象收到不同消息时产生不同的动作. 这里先说下为什么会用到虚函数: 以下面的程序为例: 这个程序中,Carp是Fish的继承类,而Carp中覆盖了Swim这 ...

  8. C++中虚函数、虚指针和虚表详解

    关于虚函数的背景知识 用virtual关键字申明的函数叫做虚函数,虚函数肯定是类的成员函数. 存在虚函数的类都有一个一维的虚函数表叫做虚表.每一个类的对象都有一个指向虚表开始的虚指针.虚表是和类对应的 ...

  9. 虚函数,虚指针和虚表详解

    关于虚函数的背景知识 1. 用virtual关键字申明的函数叫做虚函数,虚函数肯定是类的成员函数. 2. 存在虚函数的类都有一个一维的虚函数表叫做虚表.每一个类的对象都有一个指向虚表开始的虚指针.虚表 ...

最新文章

  1. matlab sae模型,matlab的Deep Learning的toolbox 中的SAE算法
  2. Ubuntu 安装 Linux Deepin 截图工具(.deb)
  3. 融合存储超越统一存储
  4. android 重新打开活动,android – RxJava在活动恢复后重新订阅事件
  5. UNIX环境高级编程 第7章 进程环境
  6. 英语阅读计算机病毒是指,阅读计算机病毒大战的答案
  7. VS2012更改/重置默认开发环境
  8. 四旋翼无人机飞控系统设计(输出分配)
  9. 联盟链之hyperledger-fabric
  10. 算法设计-天下会评选一级弟子
  11. macOS 13 Ventura系统自动开机在哪设置?
  12. 吴恩达课程作业中的lr_utils下载
  13. 客户沟通的方式:礼貌待客沟通方式,技巧推广沟通方式,个性服务沟通方式
  14. 操作系统——吸烟者问题
  15. 分布式系统的特点及问题
  16. Linux桌面 失败,ubuntu无法进入桌面,安装ubuntu-desktop失败的解决办法
  17. HTTP Digest Authentication 使用心得
  18. 凌晨三点半,见证中国奥运体育代表团的第30和31块金牌!
  19. 0.96寸OLED(SSD1306)屏幕显示(一)——基础功能介绍
  20. LDK3读书笔记(第一章:LINUX内核简史)

热门文章

  1. win7 docker的受难记——exit status 255的终极解决
  2. 软件测试工程师发展方向,主要有哪些?
  3. 认识一下CCSA(中国通信标准化协会)
  4. Java树形结构模糊搜索,模糊匹配
  5. HIVE判断字符串是否是数字
  6. [每日一氵]求解一阶线性常系数微分方程组
  7. 微信小程序 页面传值文本解密问题
  8. [IT 男人帮 11-11] 毕业一年写给自己的警戒书-褪墨
  9. 【ICPC模板】卡迈克尔函数
  10. 还原精灵的另类卸载方法