美文网首页
虚函数表

虚函数表

作者: 动静之逸 | 来源:发表于2018-08-23 01:57 被阅读0次

为什么有虚函数表?

主要还是为了实现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

相关文章

  • Boolan_C++面向对象高级编程(下)_第二周笔记

    1、 对象模型:关于vptr(虚函数表指针)和vtbl(虚函数表) 1.1 课堂内容总结 (1)class中有虚函...

  • (Boolan) 面向对象高级编程(下)第四周笔记

    一、虚函数表 对C++ 虚函数(Virtual Function)是通过一张虚函数表(Virtual Table)...

  • 十七、多态(二)

    1. 多态的实现原理 1.1虚函数表和vptr指针 当类中声明虚函数时,编译器会在类中生成一个虚函数表; 虚函数表...

  • C++虚函数表(多态的实现)

    多态是C++的三大特性之一,是通过虚函数表来实现的。关于虚函数表: 每个含有虚函数的类都有一张虚函数表(vtbl)...

  • (Boolan) 面向对象高级编程(下)第四周笔记

    一、虚函数表 对C++虚函数(Virtual Function)是通过一张虚函数表(Virtual Table)来...

  • 查漏补缺

    C++虚函数: 多态: 静态多态(重载)、动态多态(虚函数) 虚函数 虚函数表:编译器为每个类创建了一个虚函数表...

  • C++对象模型(2)

    本文预览: 关于vptr(虚函数表指针)和vtbl(虚函数表) 关于this指针 关于Dynamic Bindin...

  • 虚函数表

    虚函数表对C++ 了解的人都应该知道虚函数(Virtual Function)是通过一张虚函数表(Virtual ...

  • 虚函数表

    为什么有虚函数表? 主要还是为了实现C++的多态,即运行时绑定,而且虚函数表是在编译时期就已确认,后续链接、执行时...

  • 虚函数表

    代码 先不说废话啦,上代码: 输出: 结论: 多重继承会有多个虚函数表,几重继承,就会有几个虚函数表。这些表按照派...

网友评论

      本文标题:虚函数表

      本文链接:https://www.haomeiwen.com/subject/umiuiftx.html