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

本文详细介绍了C++中的虚函数和多态性原理。每个包含虚函数的类都有一个虚表,其中存储了虚函数的地址。对象包含一个指向虚表的指针,确保在运行时能调用正确的函数。在构造函数中初始化虚表指针,实现动态绑定。虚函数调用时,对象的虚表指针决定了调用哪个函数,从而实现多态。纯虚函数和抽象类的概念也在此文中提及。
摘要由CSDN通过智能技术生成

关于虚函数的背景知识

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

详细原理:

编译器在编译的时候,发现Base类中有虚函数,此时编译器会为每个包含虚函数的类创建一个虚表(即vtable),该表是一个一维数组(而不是一个链表),在这个数组中存放每个虚函数的地址。由于Base类和Derive类都包含了一个虚函数func(),编译器会为这两个类都建立一个虚表。

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

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

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

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

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

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

总结(基类有虚函数的情况下):

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

原文链接:https://blog.csdn.net/qq_34484472/article/details/78232465

虚函数(virtual function)是用于实现多态的一种机制,它允许通过基类的指针或引用调用派生类的方法。 虚函数的原理: 在C++虚函数的实现原理是通过虚函数表(vtable)和虚函数指针(vptr)来实现的。 1. 虚函数表(vtable):每个包含虚函数的类都有一个虚函数表,其存储了该类所有虚函数地址虚函数表是一个静态的数据结构,它在编译时生成并与类的类型相关联。每个类只有一个对应的虚函数表。 2. 虚函数指针(vptr):每个对象在内存都有一个隐藏的指针,称为虚函数指针(vptr),它指向该对象所属类的虚函数表。虚函数指针存在于对象的内存布局,但是对于程序员是透明的,无法直接访问。 当通过基类指针或引用调用虚函数时,编译器会根据对象的实际类型来确定调用哪个派生类的方法。这个过程主要依靠虚函数指针虚函数表来实现: 1. 使用基类指针或引用调用虚函数时,编译器会根据指针或引用的静态类型(基类类型)找到虚函数表的地址。 2. 通过虚函数表的地址,可以访问到虚函数表,根据偏移量找到对应的虚函数地址。 3. 最终,通过虚函数地址调用相应的派生类的虚函数。 需要注意的是,虚函数的调用是在运行时动态决定的,而不是在编译时确定的。这样可以实现多态性,让程序能够根据对象的实际类型来调用相应的函数。 总的来说,虚函数通过虚函数表和虚函数指针来实现动态绑定,使得通过基类指针或引用调用虚函数时能够根据对象的实际类型来确定调用哪个派生类的方法,实现多态性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值