虚函数表

来源:互联网 发布:mac 10.11 替换 matlab 编辑:程序博客网 时间:2024/05/21 21:50

虚函数表

@(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: 0x400d00Base::fBase::gBase::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: 0x401160Base::fBase::gBase::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: 0x401130Derive::fBase::gBase::hDerive::g1Derive::h1

可以看到,继承的时候,父类的虚函数在虚函数表的前面,子类的虚函数在虚函数的后面,子类中覆盖的虚函数,会在父类对应的虚函数的地方覆盖。

另外,子类继承的,没有修改的虚函数,与父类的虚函数不是同一个地址。

0 0