虚函数表再学习
来源:互联网 发布:华迈摄像机软件 编辑:程序博客网 时间:2024/06/05 13:28
测试环境:编译器gcc,环境WIN7,64位系统
参考:http://blog.csdn.net/haoel/article/details/1948051
结论:1. 虚函数指针存储在类其他成员之前
2. 基类的虚函数表、子类的虚函数表是分别存放的
3. 子类虚函数表中存储的内容依次为:继承自基类的虚函数(按声明顺序存放)、子类自己的虚函数(按声明顺序存放);
4. 即使子类没有自己的虚函数,他仍然会创建一个虚函数指针以及虚函数表,虚函数表内容与父类相同,但存放地址不同。
5. 当有覆盖发生时,子类虚函数表更新,基类不变。
首先是验证代码:
#include <iostream>using namespace std;class A { virtual void a(){cout << "A:a" <<endl;} virtual void b(){cout << "A:b" <<endl;} virtual void c(){cout << "A:c" <<endl;}};class AA :public A { void a(){cout << "AA:a" <<endl;} virtual void aa(){cout << "AA:aa" <<endl;} virtual void bb(){cout << "AA:bb" <<endl;}};class AAA : public AA {};int main(){ typedef void (*Fun)(void); A a ; AA aa ; AAA aaa ; Fun pFun = NULL; cout<<"address: "<< (int*)(&a) << endl; cout<<"ap: "<< (int*)*(int*)(&a) << endl; cout<<"address: "<< &aa << endl; cout<<"aap: "<< (int*)*(int*)(&aa) << endl; cout<<"address: "<< &aaa << endl; cout<<"aap: "<< (int*)*(int*)(&aaa) << endl; pFun = (Fun)*(int*)*(int*)(&a); pFun(); pFun = (Fun)*((int*)*(int*)(&a)+1); pFun(); pFun = (Fun)*((int*)*(int*)(&a)+2); pFun(); pFun = (Fun)*(int*)*(int*)(&aa); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+1); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+2); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+3); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+4); pFun(); pFun = (Fun)*(int*)*(int*)(&aaa); pFun(); cout<<"sizeof A:" << sizeof(A) <<endl; cout<<"sizeof AA:" << sizeof(AA) <<endl; cout<<"sizeof AAA:" << sizeof(AAA) <<endl; cout<<"sizeof a:" << sizeof(a) <<endl; cout<<"sizeof aa:" << sizeof(aa) <<endl; cout<<"sizeof aaa:" << sizeof(aaa) <<endl;}
验证:
结论一:虚函数指针存储在类其他成员之前
看这里:
cout<<"address: "<< &aa << endl; cout<<"aap: "<< (int*)*(int*)(&aa) << endl; .... pFun = (Fun)*(int*)*(int*)(&aa); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+1); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+2); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+3); pFun(); pFun = (Fun)*((int*)*(int*)(&aa)+4); pFun();
获取虚函数表地址的方式是:找到对象的首地址,获取该地址内存放的地址。在验证之初假设虚函数指针在类的最前面,代码运行正常,得证。
结论二:基类的虚函数表、子类的虚函数表是分别存放的
ap、aap、aaap分别代表基类、子类、孙子类的虚函数表地址,得证;
结论三:子类虚函数表中存储的内容依次为:继承自基类的虚函数(按声明顺序存放)、子类自己的虚函数(按声明顺序存放);
将子类中覆盖基类的a()方法注释掉,运行截图如下:
框内为子类虚函数表的存放内容:依次是基类的三个方法按声明顺序存放,子类两个方法按声明顺序存放;
结论四:即使子类没有自己的虚函数,他仍然会创建一个虚函数指针以及虚函数表,虚函数表内容与父类相同,但存放地址不同。
将子类代码修改为,实际就是注释掉所有自己的方法:
class AA :public A { //void a(){cout << "AA:a" <<endl;} //virtual void aa(){cout << "AA:aa" <<endl;} //virtual void bb(){cout << "AA:bb" <<endl;}};再次运行结果:
看橙色框内的内容:地址是不相同的,但打印出来的内容是相同的(无覆盖)。
结论五:当有覆盖发生时,子类虚函数表更新,基类不变。
修改子类代码:
class AA :public A { void a(){cout << "AA:a" <<endl;} virtual void aa(){cout << "AA:aa" <<endl;} virtual void bb(){cout << "AA:bb" <<endl;}};运行结果:
父类的虚函数表内容没有改变,子类虚函数表原本为基类的a( )方法,替换为了子类的a( )方法。
以上五个结论得到验证。
0 0
- 虚函数表再学习
- 学习虚函数表
- C++学习笔记-虚函数及虚函数表
- 虚函数学习之虚函数表--四
- c++学习-虚函数学习
- C++虚函数表学习笔记
- C++虚函数表调用学习
- 学习笔记_虚函数表
- 函数指针再学习
- 虚函数学习总结
- 虚函数学习笔记
- 虚函数的学习
- 虚函数学习2
- 虚函数学习
- 攻击虚函数学习
- 新手学习----虚函数-
- 虚函数学习
- 虚函数学习
- android studio 考呗.so文件运行报错 解放方案
- C语言文件操作
- [置顶]Labeling Balls--poj3687
- redis 写磁盘出错 Can’t save in background: fork: Cannot allocate memory (转)
- 自定义控件其实很简单1/3
- 虚函数表再学习
- 女子自称教师逼小学生让座 将其书包扔出窗外
- 第三方登录(ShareSDK)
- 时分秒选择控件
- POJ 2761(求区间第k小值)
- Android USB相关广播的使用
- C#窗体中根据两点坐标绘制直线
- subst
- UItableView中编辑,删除,移动学习笔记