时间:2021-05-20
虚函数
基类中使用virtual关键字声明的函数,称为虚函数。
虚函数的实现,通过虚函数表来实现的。即V-table 这个表中有一个类,用于储存虚函数的地址。解决其继承,覆盖的问题,用于保证其真实反映的函数。这样有虚函数的实例,将会储存在这个实例的内存中。即用父类的指针,操作子类的时候,通过虚函数表来实现找寻到父类。
定义下方的一个类
class Base{public: virtual void f(){ cout << "Base::f" << endl; } virtual void g(){ cout << "Base::g" << endl; } virtual void h(){ cout << "Base::h" << endl; }}然后通过实例化得到虚函数表
Base* b = new Base(); //Fun pFun = NULL; cout << "虚函数表地址 " << (int*)(&(*b)) << endl; // 强制转换成为指针类型,然后输出 cout << "虚函数表 - 第一个函数地址" << (int*)*(int*)(&(*b)) << endl; // 取虚函数的地址,然后获得虚函数的头指向的第一个存储函数的内存空间。然后获取第一个函数的内存地址然后查看输出的结果
虚函数表地址 0xb31268
虚函数表 - 第一个函数地址0x4c1490
根据地址可以看到,指针b指向创建的实例的地址,其首地址储存着虚函数表的地址,然后再次通过指针访问,得到虚函数表的第一个函数的指针的地址。
一般继承,没有虚函数覆盖
在上方的继承中,子类没有重载任何父类的函数,那么在虚函数列表中,就代表着
虚函数按照声明的顺序放入表中。
父类的虚函数,在子类的虚函数的前面。
一般继承,有虚函数覆盖
此时内存中地址如下
注意,最重要的一点是,地址上,覆盖的f()函数,被放置到了父类的f()函数上。于是就可以有下方的程序
Base *b = new Derive(); b->f();使用一个类型为Base的指针b指向一个新建的实例Derive(),此时对于指针b指向的虚函数表中,此时f()的地址被Devieive函数的地址被覆盖, 形成如上的虚函数表。
此时访问成功
多重继承 无虚函数的覆盖
假设有如上的内容
此时虚函数表
每个父类都有自己的虚表,子类的成员函数被放到第一个父类的表中。即第一个父类的表是按照声明的顺序来判断。
多重继承 有虚函数覆盖
此时表如下
此时父类的被替换了。
这样就实现了多重继承,代码如下
Derive d; Base1 *b1 = &d; Base2 *b2 = &d; Base3 *b3 = &d; b1->f(); b2->f(); b3->f();此时代码如上。
纯虚函数
virtual int area() = 0;实现一个纯虚函数,此时可以在派生类中更好的定义纯虚函数。
以上就是C++ 虚函数专题的详细内容,更多关于C++ 虚函数的资料请关注其它相关文章!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
1、C++对象的内存分布和虚函数表:C++对象的内存分布和虚函数表注意,对象中保存的是虚函数表指针,而不是虚函数表,虚函数表在编译阶段就已经生成,同类的不同对象
探讨C++虚函数virtual有无虚函数的对比C++中的虚函数用于解决动态多态问题,虚函数的作用是允许在派生类中重新定义与积累同名的函数,并且可以通过基类指针或
前言c++分为编译时多态和运行时多态。运行时多态依赖于虚函数,大部分人或许听说过虚函数是由虚函数表+虚函数指针实现的,但,真的是这样吗?虽然c++规范有着复杂的
前言自上一个帖子之间跳过了一篇总结性的帖子,之后再发,今天主要研究了c++语言当中虚函数对多态的实现,感叹于c++设计者的精妙绝伦c++中虚函数表的作用主要是实
本文针对C++的虚函数的实现机制进行较为深入的分析,具体如下:1、简单地说,虚函数是通过虚函数表实现的。那么,什么是虚函数表呢?事实上,如果一个类中含有虚函数,