动态联编学习:三、虚函数表VTABLE

本文深入探讨C++中虚函数的工作原理,包括虚拟函数表(VTABLE)和vptr的概念,以及它们如何支持动态联编,实现多态性。通过具体代码示例解释编译器如何处理虚函数,展示不同情况下函数调用的过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

      编译器在执行过程中遇到virtual关键字的时候,将自动安装动态联编需要的机制,首先为这些包含virtual函数的类(注意不是类的实例)--即使是祖先类包含虚函数而本身没有--建立一张虚拟函数表VTABLE。在这些虚拟函数表中,编译器将依次按照函数声明次序放置类的特定虚函数的地址。同时在每个带有虚函数的类中放置一个称之为vpointer的指针,简称vptr,这个指针指向这个类的VTABLE

      关于虚拟函数表,有几点必须声明清楚:

1. 每一个类别只能有一个虚拟函数表,如果该类没有虚拟函数,则不存在虚拟函数表。

2. C++编译时候编译器会在含有虚函数的类中加上一个指向虚拟函数表的指针vptr

3. 从一个类别诞生的每一个对象,将获取该类别中的vptr指针,这个指针同样指向类的VTABLE

因此类、对象、VTABLE的层次结构可以用下图表示。其中X类和Y类的对象的指针都指向了X,Y的虚拟函数表,同时X,Y类自身也包含了指向虚拟函数的指针。

 

      编译器在编译上面这段代码的时候将为这shapecircle两个对象分别建立一个VTABLE表,这些表依次填充派生类对象基类对象中声明的所有的虚函数地址。如果派生类本身没有重新定义基类的虚函数,那么填充的就是基类的虚函数地址。这样一旦如果函数调用一个派生类不存在的方法时候能够自动调用基类方法。然后编译器在每个类中放置一个vptr,一般置于对象的起始位置,继而在对象的构造函数中将vptr初始化为本类的VTABLE的地址。

 

C++ 编译程序时候按下面的步骤进行工作:

1)为各类建立虚拟函数表,如果没有虚函数则不建立。

2)暂时不连接虚函数,而是将各个虚函数的地址放入虚拟函数表中。

3)直接连接各静态函数。

 

 

      执行时候,诞生了oneshapecircleshape两个对象,oneshape对象的vptr指针指向shapeVTABLEcircleshape对象的vptr指针指向circleshapeVTABLE,在执行oneshape.fun()的时候,fun函数的this指针指向了oneshape对象,进入fun()之后程序继续执行this->draw(),由于this指向oneshape对象,oneshapevptr又指向shape类的VTABLE,这样就从VTABLE中得到需要绑定的函数的地址,并连接起来。同样,this-> area()也经由oneshape对象而连接到相应的函数上,如图。

 

      现在我们执行baseshape.fun()。

 

      函数进入fun函数之后,函数的this指针将指向basefun对象,另一方面basefun指向一个circleshape,因此this指针指向的实际上为circleshape对象,而circleshapevptr指针指向circle类的虚拟函数表,这样编译器将从虚拟表中取出circle::draw()circle::area()的地址,进行连接。因为circle本身没有重新定义area()方法,因此编译器使用shapearea()方法。如图:

 

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值