• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号
您的位置:首页 > 程序设计 >C语言 > 浅谈C++中虚函数实现原理揭秘

浅谈C++中虚函数实现原理揭秘

作者: 字体:[增加 减小] 来源:互联网 时间:2017-05-28

通过本文主要向大家介绍了浅谈二次函数,浅谈函数的一致连续性,浅谈三角函数,浅谈一次函数,浅谈室内设计原理等相关知识,希望对您有所帮助,也希望大家支持linkedu.com www.linkedu.com

编译器到底做了什么实现的虚函数的晚绑定呢?我们来探个究竟。     

编译器对每个包含虚函数的类创建一个表(称为V TA B L E)。在V TA B L E中,编译器放置特定类的虚函数地址。在每个带有虚函数的类 中,编译器秘密地置一指针,称为v p o i n t e r(缩写为V P T R),指向这个对象的V TA B L E。通过基类指针做虚函数调 用时(也就是做多态调用时),编译器静态地插入取得这个V P T R,并在V TA B L E表中查找函数地址的代码,这样就能调用正确的函数使晚捆绑发生。为每个类设置V TA B L E、初始化V P T R、为虚函数调用插入代码,所有这些都是自动发生的,所以我们不必担心这些。利用虚函数, 这个对象的合适的函数就能被调用,哪怕在编译器还不知道这个对象的特定类型的情况下。(《C++编程思想》)

在任何类中不存在显示的类型信息,可对象中必须存放类信息,否则类型不可能在运行时建立。那这个类信息是什么呢?

我们来看下面几个类:

class no_virtual

{
public:
  void fun1() const{}
  int fun2() const { return a; }
private:
  int a;
} class one_virtual
{
public:
  virtual void fun1() const{}
  int fun2() const { return a; }
private:
  int a;
} class two_virtual
{
public:
  virtual void fun1() const{}
  virtual int fun2() const { return a; }
private:
  int a;
} 
</div>

以上三个类中:

no_virtual没有虚函数,sizeof(no_virtual)=4,

类no_virtual的长度就是其成员变量整型a的长度;

one_virtual有一个虚函数,sizeof(one_virtual)=8;

two_virtual 有两个虚函数,sizeof(two_virtual)=8;

有一个虚函数和两个虚函数的类的长度没有区别,其实它们的长度就是no_virtual的 长度加一个void指针的长度,它反映出,如果有一个或多个虚函数,编译器在这个结构中插入一个指针( V P T R)。在one_virtual 和 two_virtual之间没有区别。这是因为V P T R指向一个存放地址的表,只需要一个指针,因为所有虚函数地址都包含在这个表中。 这个VPTR就可以看作类的类型信息。

那我们来看看编译器是怎么建立VPTR指向的这个虚函数表的。先看下面两个类:

class base
{
public:
  void bfun(){}
  virtual void vfun1(){}
  virtual int vfun2(){}
private:
  int a;
} class derived : public base
{
public:
  void dfun(){}
  virtual void vfun1(){}
  virtual int vfun3(){}
private:
  int b;
} 
</div>

两个类VPTR指向的虚函数表(VTABLE)分别如下:

base类

 

      ——————
VPTR——> |&base::vfun1 |
      ——————
     |&base::vfun2 |
     ——————
</div>

derived类

      ———————
VPTR——> |&derived::vfun1 |
      ———————
     |&base::vfun2 |
     ———————
     |&derived::vfun3 |
     ———————
</div>

每当创建一个包含有虚函数的类或从包含有虚函数的类派生一个类时,编译器就为这个类创建一个VTABLE,如上图所示。在这个表中,编译器放置了在这个类 中或在它的基类中所有已声明为virtual的函数的地址。如果在这个派生类中没有对在基类中声明为virtual的函数进行重新定义,编译器就使用基类 的这个虚函数地址。(在derived的VTABLE中,vfun2的入口就是这种情况。)然后编译器在这个类中放置VPTR。当使用简单继承时,对于每 个对象只有一个VPTR。VPTR必须被初始化为指向相应的VTABLE,这在构造函数中发生。

一旦VPTR被初始化为指向相应的VTABLE,对象就"知道"它自己是什么类型。但只有当虚函数被调用时这种自我认知才有用。       

个人总结如下:

1、从包含虚函数的类派生一个类时,编译器就为该类创建一个VTABLE。其每一个表项是该类的虚函数地址。

2、在定义该派生类对象时,先调用其基类的构造函数,然后再初始化VPTR,最后再调用派生类的构造函数( 从二进制的视野来看,所谓基类子类是一个大结构体,其中this指针开头的四个字节存放虚函数表头指针。执行子类的构造函数的时候,首先调用基类构造函数,this指针作为参数,在基类构造函数中填入基类的vptr,然后回到子类的构造函数,填入子类的vptr,覆盖基类填入的vptr。如此以来完成vptr的初始化。 )

3、在实现动态绑定时,不能直接采用类对象,而一定要采用指针或者引用。因为采用类对象传值方式,有临时基类对象的产生,而采用指针,则是通过指针来访问外部的派生类对象的VPTR来达到访问派生类虚函数的结果。       

VPTR 常常位于对象的开头,编译器能很容易地取到VPTR的值,从而确定VTABLE的位置。VPTR总指向VTABLE的开始地址,所有基类和它的子类的虚函 数地址(子类自己定义的虚函数除外)在VTABLE中存储的位置总是相同的,如上面base类和derived类的VTABLE中vfun1和vfun2 的地址总是按相同的顺序存储。编译器知道vfun1位于VPTR处,vfun2位于VPTR+1处,因此在用基类指针调用虚函数时,编译器首先获取指针指 向对象的类型信息(VPTR),然后就去调用虚函数。如一个base类指针pBase指向了一个derived对象,那pBase->vfun2 ()被编译器翻译为 VPTR+1 的调用,因为虚函数vfun2的地址在VTABLE中位于索引为1的位置上。同理,pBase->vfun3 ()被编译器翻译为 VPTR+2的调用。这就是所谓的晚绑定。

以上这篇浅谈C++中虚函数实现原理揭秘就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。

</div>
分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • 浅谈返回函数内部new分配的内存的引用
  • 浅谈在函数中返回动态的内存
  • 浅谈C++中虚函数实现原理揭秘
  • 浅谈C++中的构造函数分类及调用规则

相关文章

  • 2017-05-28C++实现随机生成迷宫地牢
  • 2017-05-28c++中typename和class的区别介绍
  • 2017-05-28老生常谈C++中实参形参的传递问题
  • 2017-05-28老生常谈C语言静态函数库的制作和使用
  • 2022-04-30C语言编译器(C语言编程软件)完全攻略(包含所有平台)
  • 2017-05-28简单介绍C++编程中派生类的析构函数
  • 2017-05-28C语言编程中从密码文件获取数据的函数总结
  • 2017-05-28C++双向循环列表用法实例
  • 2017-05-28基于C/C++时间函数的使用详解
  • 2017-12-08Oil Deposits

文章分类

  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号

最近更新的内容

    • C++ 单链表的基本操作(详解)
    • C++中sting类的简单实现方法
    • C++ namespace相关语法实例分析
    • c++中拷贝构造函数的参数类型必须是引用
    • C++不使用变量求字符串长度strlen函数的实现方法
    • C++双向循环列表用法实例
    • C 语言基础教程(我的C之旅开始了)[六]
    • 用C++面向对象的方式动态加载so的方法
    • C++程序检测内存泄漏的方法分享
    • 详解约瑟夫环问题及其相关的C语言算法实现

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有