虚函数表

虚函数表

@(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

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

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值