- 当类中包含虚函数时,则该类每个对象中在内存分配中除去数据外还包含了一个虚函数表指针(vfptr),指向虚函数表(vftable),虚函数表中存放了该类包含的虚函数的地址。
- 当子类通过虚继承的方式从父类中派生出来,此时称父类为子类的虚基类。子类中将包含虚基表指针(vbptr),指向虚基类表(vbtable)
- 在单继承形式下,子类将完全获得父类的虚函数表和数据(假入父类中有虚函数的话)。如果子类中重写了父类的虚函数,就会在虚函数表中原本记录父类中虚函数的地址覆盖为子类中对应的重定义后的该函数地址,否则不做变动。如果在子类中定义了新的虚函数,则虚函数表中会追加一条记录,记录该函数的地址(虚函数表中是顺序存放虚函数地址的,记住,虽然虚函数表中添加了内容,但是此时对于该类的大小来说并未发生改变,因为始终只有一个指向虚函数表的指针vfptr)。
- 如果在派生类对象直接访问自身中的重定义的虚函数是不会触发多态机制的,因为这个函数调用在编译时期是可以确定的,编译器只需要直接调用即可;
- 当对父类指针赋予不同的子类指针时,在调用子类中重定义的虚函数时才会触发多态机制,即动态的在运行期间调用属于子类的该函数。(可适度地了解下覆盖override和重载overload的区别)
- 在多重继承形式下,派生类会把所有的父类按继承时的顺序包含在自身内部。每个父类对应一个单独的虚函数表。多重继承下,子类不再具有自身的虚函数表,它的虚函数表与第一个父类的虚函数表合并了。同样的,如果子类重写了任意父类的虚函数,都会覆盖对应的函数地址记录。如果如果两个虚基类都有该函数那么两个虚函数表的记录都需要被覆盖!
- 此时由于多重继承下将存在对公共基类的多份拷贝问题,为了节省内存空间,故而出现了虚拟继承,那么将只生成一个共享的基类。
- 虚继承的引入把对象的模型变得复杂,除了每个基类和公共基类的虚函数表指针vfptr需要记录外,每个虚拟继承了的父类还需要记录一个虚基类表vbtable的指针vbptr。
- 虚基类表每项记录了被继承的虚基类子对象相对于虚基类表指针的偏
来自:https://blog.csdn.net/u012209626/article/details/48682555
虚函数开销
- 每一个包含虚函数的类都需要专门的空间存放这个类的虚函数表。
- 需要在包含虚函数的类的每个对象中放置一个额外的指针。
- 必须放弃内联
特性 | 增加对象大小 | 增加类的大小 | 减少内联 |
虚函数 | 是 | 是 | 是 |
多继承 | 是 | 是 | 否 |
虚基类 | 经常 | 有时 | 否 |
RTTI | 否 | 是 | 否 |
虚函数与虚继承的具体实现
一、基本对象模型
class MyClass { int var; public: virtual void fun() {} };
编译输出的MyClass对象结构如下:
1> class MyClass size(8): 1> +--- 1> 0 | {vfptr} 1> 4 | var 1> +--- 1> 1> MyClass::$vftable@: 1> | &MyClass_meta 1> | 0 1> 0 | &MyClass::fun 1> 1> MyClass::fun this adjustor: 0
从这段信息中我们看出,MyClass对象大小是8个字节。前四个字节存储的是虚函数表的指针vfptr,后四个字节存储对象成员var的值。虚函数表的大小为4字节,就一条函数地址,即虚函数fun的地址,它在虚函数表vftable的偏移是0。
因此,MyClass对象模型的结果如图1所示
图1 MyClass对象模型
MyClass的虚函数表虽然只有一条函数记录,但是它的结尾处是由4字节的0作为结束标记的。
adjust表示虚函数机制执行时,this指针的调整量,假如fun被多态调用的话,那么它的形式如下:
*(this+0)[0]()
总结虚函数调用形式,应该是:
*(this指针+调整量)[虚函数在vftable内的偏移]()
二、单重继承对象模型
我们定义一个继承于MyClass类的子类MyClassA,它重写了fun函数,并且提供了一个新的虚函数funA。
class MyClassA:public MyClass { int varA; public: virtual void fun() {} virtual void funA() {} };
它的对象模型为:
1> class MyClassA size(12): 1> +--- 1> | +--- (base class MyClass) 1> 0 | | {vfptr} 1> 4 | | var 1> | +--- 1> 8 | varA 1> +--- 1> 1> MyClassA::$vftable@: 1> | &MyClassA_meta 1> | 0 1> 0 | &MyClassA::fun 1> 1 | &MyClassA::funA 1> 1> MyClassA::fun this adjustor: 0 1> MyClassA::funA this adjustor: 0
可以看出,MyClassA将基类MyClass完全包含在自己内部,包括vfptr和var。并且虚函数表内的记录多了一条——MyClassA自己定义的虚函数funA。它的对象模型如图2所示。
图2 MyClassA对象模型
我们可以得出结论:在单继承形式下,子类的完全获得父类的虚函数表和数据。子类如果重写了父类的虚函数(如fun),就会把虚函数表原本fun对应的记录(内容MyClass::fun)覆盖为新的函数地址(内容MyClassA::fun),否则继续保持原本的函数地址记录。如果子类定义了新的虚函数,虚函数表内会追加一条记录,记录该函数的地址(如MyClassA::funA)。
使用这种方式,就可以实现多态的特性。假设我们使用如下语句:
MyClass*pc=new MyClassA; pc->fun();
编译器在处理第二条语句时,发现这是一个多态的调用,那么就会按照上边我们对虚函数的多态访问机制调用函数fun。
*(pc+0)[0]()
因为虚函数表内的函数地址已经被子类重写的fun函数地址覆盖了,因此该处调用的函数正是MyClassA::fun,而不是基类的MyClass::fun。
如果使用MyClassA对象直接访问fun,则不会出发多态机制,因为这个函数调用在编译时期是可以确定的,编译器只需要直接调用MyClassA::fun即可。
三、多重继承对象模型
和前边MyClassA类似,我们也定义一个类MyClassB
class MyClassB:public MyClass { int varB; public: virtual void fun() {} virtual void funB() {} };
它的对象模型和MyClassA完全类似,这里就不再赘述了。
为了实现多重继承,我们再定义一个类MyClassC。
class MyClassC:public MyClassA,public MyClassB { int varC; public: virtual void funB() {} virtual void funC() {} };
为了简化,我们让MyClassC只重写父类MyClassB的虚函数funB,它的对象模型如下:
1> class MyClassC size(28): 1> +--- 1> | +--- (base class MyClassA) 1> | | +--- (base class MyClass) 1> 0 | | | {vfptr} 1> 4 | | | var 1> | | +--- 1> 8 | | varA 1> | +--- 1> | +--- (base class MyClassB) 1> | | +--- (base class MyClass) 1> 12 | | | {vfptr} 1> 16 | | | var 1> | | +--- 1> 20 | | varB 1> | +--- 1> 24 | varC 1> +--- 1> 1> MyClassC::$vftable@MyClassA@: 1> | &MyClassC_meta 1> | 0 1> 0 | &MyClassA::fun 1> 1 | &MyClassA::funA 1> 2 | &MyClassC::funC 1> 1> MyClassC::$vftable@MyClassB@: 1> | -12 1> 0 | &MyClassB::fun 1> 1 | &MyClassC::funB 1> 1> MyClassC::funB this adjustor: 12 1> MyClassC::funC this adjustor: 0
上述红色数据表示重复数据,这也是钻石继承造成的问题---代码冗余和二义性
和单重继承类似,多重继承时MyClassC会把所有的父类全部按序包含在自身内部。而且每一个父类都对应一个单独的虚函数表。MyClassC的对象模型如图3所示。
图3 MyClassC对象模型
多重继承下,子类不再具有自身的虚函数表,它的虚函数表与第一个父类的虚函数表合并了。同样的,如果子类重写了任意父类的虚函数,都会覆盖对应的函数地址记录。如果MyClassC重写了fun函数(两个父类都有该函数),那么两个虚函数表的记录都需要被覆盖!在这里我们发现MyClassC::funB的函数对应的adjust值是12,按照我们前边的规则,可以发现该函数的多态调用形式为:
*(this+12)[1]()
此处的调整量12正好是MyClassB的vfptr在MyClassC对象内的偏移量。
虚函数表,虚函数继承实现:https://blog.csdn.net/best_fiends_zxh/article/details/59111761
虚拟继承是为了解决多重继承下公共基类的多份拷贝问题。比如上边的例子中MyClassC的对象内包含MyClassA和MyClassB子对象,但是MyClassA和MyClassB内含有共同的基类MyClass。为了消除MyClass子对象的多份存在,我们需要让MyClassA和MyClassB都虚拟继承于MyClass,然后再让MyClassC多重继承于这两个父类。相对于上边的例子,类内的设计不做任何改动,先修改MyClassA和MyClassB的继承方式:
class MyClassA:virtual public MyClass class MyClassB:virtual public MyClass class MyClassC:public MyClassA,public MyClassB
由于虚继承的本身语义,MyClassC内必须重写fun函数,因此我们需要再重写fun函数。这种情况下,MyClassC的对象模型如下:
1> class MyClassC size(36): 1> +--- 1> | +--- (base class MyClassA) 1> 0 | | {vfptr} 1> 4 | | {vbptr} 1> 8 | | varA 1> | +--- 1> | +--- (base class MyClassB) 1> 12 | | {vfptr} 1> 16 | | {vbptr} 1> 20 | | varB 1> | +--- 1> 24 | varC 1> +--- 1> +--- (virtual base MyClass) 1> 28 | {vfptr} 1> 32 | var 1> +--- 1> 1> MyClassC::$vftable@MyClassA@: 1> | &MyClassC_meta 1> | 0 1> 0 | &MyClassA::funA 1> 1 | &MyClassC::funC 1> 1> MyClassC::$vftable@MyClassB@: 1> | -12 1> 0 | &MyClassC::funB 1> 1> MyClassC::$vbtable@MyClassA@: 1> 0 | -4 1> 1 | 24 (MyClassCd(MyClassA+4)MyClass) 1> 1> MyClassC::$vbtable@MyClassB@: 1> 0 | -4 1> 1 | 12 (MyClassCd(MyClassB+4)MyClass) 1> 1> MyClassC::$vftable@MyClass@: 1> | -28 1> 0 | &MyClassC::fun 1> 1> MyClassC::fun this adjustor: 28 1> MyClassC::funB this adjustor: 12 1> MyClassC::funC this adjustor: 0 1> 1> vbi: class offset o.vbptr o.vbte fVtorDisp 1> MyClass 28 4 4 0
虚继承的引入把对象的模型变得十分复杂,除了每个基类(MyClassA和MyClassB)和公共基类(MyClass)的虚函数表指针需要记录外,每个虚拟继承了MyClass的父类还需要记录一个虚基类表vbtable的指针vbptr。MyClassC的对象模型如图4所示。
图4 MyClassC对象模型
注意:对于在对象中存取虚基类的问题,虚基类表仅是Microsoft编译器的解决办法。在其他编译器中,一般采用在虚函数表中放置虚基类的偏移量的方式。
一般编译器实现动态多态方法:
1、通过vbptr找到对象的vtbl(this指针的调整量);
2、找到vfptr中对应函数的指针(虚函数表中记录的个数);
3、调用对象中指针指向的函数
*(this指针+调整量)[虚函数在vftable内的偏移]()
MyClassC中的fun()函数直接继承与基类!!!
虚基类表每项记录了被继承的虚基类子对象相对于虚基类表指针的偏移量。比如MyClassA的虚基类表第二项记录值为24,正是MyClass::vfptr相对于MyClassA::vbptr的偏移量,同理MyClassB的虚基类表第二项记录值12也正是MyClass::vfptr相对于MyClassA::vbptr的偏移量。
和虚函数表不同的是,虚基类表的第一项记录着当前子对象相对与虚基类表指针的偏移。MyClassA和MyClassB子对象内的虚表指针都是存储在相对于自身的4字节偏移处,因此该值是-4。假定MyClassA和MyClassC或者MyClassB内没有定义新的虚函数,即不会产生虚函数表,那么虚基类表第一项字段的值应该是0。
通过以上的对象组织形式,编译器解决了公共虚基类的多份拷贝的问题。通过每个父类的虚基类表指针,都能找到被公共使用的虚基类的子对象的位置,并依次访问虚基类子对象的数据。至于虚基类定义的虚函数,它和其他的虚函数的访问形式相同,本例中,如果使用虚基类指针MyClass*pc访问MyClassC对象的fun,将会被转化为如下形式:
*(pc+28)[0]()
通过以上的描述,我们基本认清了C++的对象模型。尤其是在多重、虚拟继承下的复杂结构。通过这些真实的例子,使得我们认清C++内class的本质,以此指导我们更好的书写我们的程序。本文从对象结构的角度结合图例为大家阐述对象的基本模型,和一般描述C++虚拟机制的文章有所不同。
来自http://www.cnblogs.com/fanzhidongyzby/archive/2013/01/14/2859064.html
注意;两个函数的返回类型,参数类型,参数个数都得相同,不然就起不到多态的作用
#include<iostream> #include<cmath> using namespace std; class A { public : virtual void fun(int x) { cout << "A: " << x << endl; } }; class B :public A { public : virtual void fun(float x) { cout << "B: " << x << endl; } }; void test(A & x) { int i = 1; x.fun(i); float a = 2.0; x.fun(a); } int main() { A a; B b; test(a); test(b); return 0; }
但是有一种特殊的情况,那就是如果基类中虚函数返回一个基类指针或引用,派生类中返回一个派生类的指针或引用,则c++将其视为同名虚函数而进行迟后联编。
#include<iostream> #include<cmath> using namespace std; class A { public : virtual A * fun() { cout << "A: " << endl; return this; } }; class B :public A { public : virtual B * fun() { cout << "B: " << endl; return this; } }; void test(A & x) { x.fun(); } int main() { A a; B b; test(a); test(b); return 0; }
虚函数在g++中的实现
因为vptr明确的属于一个实例,所以vptr的赋值理应放在类的构造函数中。 g++为每个有虚函数的类在构造函数末尾中隐式的添加了为vptr赋值的操作,vtbl的生成并不是运行时的,而是在编译期就已经确定了存放在这两个地址上的,这个地址属于.rodata(只读数据段)。所以g++在编译期就为每个类确定了vtbl的内容,并且在构造函数中添加相应代码使vptr能够指向已经填好的vtbl的地址。
来自:https://www.tuicool.com/articles/iUB3Ebi
继承是如何实现的:https://blog.csdn.net/dream_1996/article/details/68931347