虚函数表

为什么有虚函数表?

主要还是为了实现C++的多态,即运行时绑定,而且虚函数表是在编译时期就已确认,后续链接、执行时都不会改变其地址

示例

#include <iostream>

using namespace std;

class Base
{
public:
    virtual void a();
    virtual void b();
    virtual void c();
};

void Base::a()
{
    cout << "base::a" << endl;
}

void Base::b()
{
    cout << "base::b" << endl;
}

void Base::c()
{
    cout << "base::c" << endl;
}

class Derived : public Base
{
public:
    void a();
};

void Derived::a()
{
    cout << "derived::a" << endl;
}

typedef void (*func)();

int main(int argc, char **argv)
{
    // 先看继承类的地址空间
    Derived derive;
    printf("derive::vptr %p\n", *(int *)&derive);

    // 获取虚函数表的地址
    int *v_ptr = (int *)(*(int *)&derive);

    printf("derive::vptr->a %p\n", *v_ptr);
    printf("derive::vptr->b %p\n", *(v_ptr+1));
    printf("derive::vptr->c %p\n", *(v_ptr+2));

    ((func)*v_ptr)();
    ((func)*(v_ptr + 1))();
    ((func)*(v_ptr + 2))();

    // 看看Base的地址空间
    Base base1;
    printf("base1::vptr %p\n", *(int *)&base1);

    // 获取虚函数表的地址
    v_ptr = (int *)(*(int *)&base1);

    printf("base1::vptr->a %p\n", *v_ptr);
    printf("base1::vptr->b %p\n", *(v_ptr + 1));
    printf("base1::vptr->c %p\n", *(v_ptr + 2));

    ((func)*v_ptr)();
    ((func)*(v_ptr + 1))();
    ((func)*(v_ptr + 2))();

    // 看看Base的地址空间
    Base base2;
    printf("base2::vptr %p\n", *(int *)&base2);

    // 获取虚函数表的地址
    v_ptr = (int *)(*(int *)&base2);

    printf("base2::vptr->a %p\n", *v_ptr);
    printf("base2::vptr->b %p\n", *(v_ptr + 1));
    printf("base2::vptr->c %p\n", *(v_ptr + 2));

    ((func)*v_ptr)();
    ((func)*(v_ptr + 1))();
    ((func)*(v_ptr + 2))();

    int test;
    cin >> test;

    return 0;
}

输出:

derive:vptr 00268B6C
derive:vptr->a 002610A0
derive:vptr->b 00261415
derive:vptr->c 002613D4
derived::a
base::b
base::c
base1:vptr 00268B34
base1:vptr->a 002610E6
base1:vptr->b 00261415
base1:vptr->c 002613D4
base::a
base::b
base::c
base2:vptr 00268B34
base2:vptr->a 002610E6
base2:vptr->b 00261415
base2:vptr->c 002613D4
base::a
base::b
base::c

结论:

  1. 虚函数和类相关,和对象无关,从base1和base2的虚函数的地址完全一样可知
  2. 继承关系时,derived重写的虚函数,会将对应虚函数表中的函数地址改写成子类的函数的地址,以实现动态binding
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 多态性:一个接口,多种方法.程序在运行时才确定调用的函数,是 oop 的核心概念. 多态性通过虚函数来实现,子类可...
    曹小恒阅读 6,518评论 2 4
  • 什么叫多态性? A:同样的消息被不同类型的对象接收时,对象会采用完全不同的行为处理消息。 多态性有几种? A:静态...
    凉拌姨妈好吃阅读 3,811评论 0 0
  • 引言 虚表是 C++ 中一个十分重要的概念,面向对象编程的多态性在 C++ 中的实现全靠虚表来实现。在聊虚表之前我...
    Cyandev阅读 19,878评论 7 38
  • //联系人:石虎QQ:1224614774昵称:嗡嘛呢叭咪哄 一、概念 1.多态是由虚函数实现的,而虚函数主要是通...
    石虎132阅读 5,608评论 0 10
  • 1、安装新立德软件包管理器http://jingyan.baidu.com/article/295430f13d2...
    hyer阅读 2,985评论 0 1