虚函数表
@(c/cpp)
参考
http://blog.csdn.net/haoel/article/details/1948051
其中,为了在64位机器上使用 ,应该把int换成long,这样才能取64位的地址。
基类
代码
#include <iostream>
#include <cstdlib>
using namespace std;
class Base {
public:
virtual void f() { cout << "Base::f" << endl; }
virtual void g() { cout << "Base::g" << endl; }
virtual void h() { cout << "Base::h" << endl; }
};
int main(void)
{
typedef void(*Fun)(void);
Base b;
Fun pFun = NULL;
cout << "虚函数表地址:(long*)(&b)" << (long*)(&b) << endl;
cout << "虚函数表 - 第一个函数地址:(long*)*(long*)(&b): " << (long*)*(long*)(&b) << endl;
cout << "虚函数表 - 第二个函数地址:(long*)*(long*)(&b)+1: " << (long*)*(long*)(&b)+1 << endl;
cout << "虚函数表 - 第三个函数地址:(long*)*(long*)(&b)+2: " << (long*)*(long*)(&b)+2 << endl;
// Invoke the first virtual function
pFun = (Fun)*((long*)*(long*)(&b));
pFun();
pFun = (Fun)*((long*)*(long*)(&b)+1); // Base::g()
pFun();
pFun = (Fun)*((long*)*(long*)(&b)+2); // Base::h()
pFun();
return 0;
}
输出:
虚函数表地址:(long*)(&b)0x7ffe79b46950
虚函数表 - 第一个函数地址:(long*)*(long*)(&b): 0x400cf0
虚函数表 - 第二个函数地址:(long*)*(long*)(&b)+1: 0x400cf8
虚函数表 - 第三个函数地址:(long*)*(long*)(&b)+2: 0x400d00
Base::f
Base::g
Base::h
多态-继承
代码
#include <iostream>
#include <cstdlib>
using namespace std;
class Base {
public:
virtual void f() { cout << "Base::f" << endl; }
virtual void g() { cout << "Base::g" << endl; }
virtual void h() { cout << "Base::h" << endl; }
};
class Derive: Base
{
virtual void f() { cout << "Derive::f" << endl; }
virtual void g1() { cout << "Derive::g1" << endl; }
virtual void h1() { cout << "Derive::h1" << endl; }
};
int main(void)
{
typedef void(*Fun)(void);
Base b;
Fun pFun = NULL;
cout << "虚函数表地址:(long*)(&b)" << (long*)(&b) << endl;
cout << "虚函数表 - 第一个函数地址:(long*)*(long*)(&b): " << (long*)*(long*)(&b) << endl;
cout << "虚函数表 - 第二个函数地址:(long*)*(long*)(&b)+1: " << (long*)*(long*)(&b)+1 << endl;
cout << "虚函数表 - 第三个函数地址:(long*)*(long*)(&b)+2: " << (long*)*(long*)(&b)+2 << endl;
// Invoke the first virtual function
pFun = (Fun)*((long*)*(long*)(&b));
pFun();
pFun = (Fun)*((long*)*(long*)(&b)+1); // Base::g()
pFun();
pFun = (Fun)*((long*)*(long*)(&b)+2); // Base::h()
pFun();
Derive d;
cout << "多态:\n虚函数表地址:(long*)(&d)" << (long*)(&d) << endl;
cout << "虚函数表 - 第1个函数地址:(long*)*(long*)(&d): " << (long*)*(long*)(&d) << endl;
cout << "虚函数表 - 第2个函数地址:(long*)*(long*)(&d)+1: " << (long*)*(long*)(&d)+1 << endl;
cout << "虚函数表 - 第3个函数地址:(long*)*(long*)(&d)+2: " << (long*)*(long*)(&d)+2 << endl;
cout << "虚函数表 - 第4个函数地址:(long*)*(long*)(&d)+3: " << (long*)*(long*)(&d)+3 << endl;
cout << "虚函数表 - 第5个函数地址:(long*)*(long*)(&d)+4: " << (long*)*(long*)(&d)+4 << endl;
// Invoke the first virtual function
pFun = (Fun)*((long*)*(long*)(&d));
pFun();
pFun = (Fun)*((long*)*(long*)(&d)+1);
pFun();
pFun = (Fun)*((long*)*(long*)(&d)+2);
pFun();
pFun = (Fun)*((long*)*(long*)(&d)+3);
pFun();
pFun = (Fun)*((long*)*(long*)(&d)+4);
pFun();
return 0;
}
输出
虚函数表地址:(long*)(&b)0x7fff83a1eed0
虚函数表 - 第一个函数地址:(long*)*(long*)(&b): 0x401150
虚函数表 - 第二个函数地址:(long*)*(long*)(&b)+1: 0x401158
虚函数表 - 第三个函数地址:(long*)*(long*)(&b)+2: 0x401160
Base::f
Base::g
Base::h
多态:
虚函数表地址:(long*)(&d)0x7fff83a1eec0
虚函数表 - 第1个函数地址:(long*)*(long*)(&d): 0x401110
虚函数表 - 第2个函数地址:(long*)*(long*)(&d)+1: 0x401118
虚函数表 - 第3个函数地址:(long*)*(long*)(&d)+2: 0x401120
虚函数表 - 第4个函数地址:(long*)*(long*)(&d)+3: 0x401128
虚函数表 - 第5个函数地址:(long*)*(long*)(&d)+4: 0x401130
Derive::f
Base::g
Base::h
Derive::g1
Derive::h1
可以看到,继承的时候,父类的虚函数在虚函数表的前面,子类的虚函数在虚函数的后面,子类中覆盖的虚函数,会在父类对应的虚函数的地方覆盖。
另外,子类继承的,没有修改的虚函数,与父类的虚函数不是同一个地址。