iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >聊一聊C++虚函数表的问题
  • 540
分享到

聊一聊C++虚函数表的问题

2024-04-02 19:04:59 540人浏览 八月长安
摘要

之前只是看过c++虚函数表相关介绍,今天有空就来写代码研究一下。 面向对象的编程语言有3大特性:封装、继承和多态。C++是面向对象的语言(与C语言主要区别),所以C++也拥有多态的特

之前只是看过c++虚函数表相关介绍,今天有空就来写代码研究一下。

面向对象编程语言有3大特性:封装、继承和多态。C++是面向对象的语言(与C语言主要区别),所以C++也拥有多态的特性。

C++中多态分为两种:静态多态动态多态

静态多态为编译器在编译期间就可以根据函数名和参数等信息确定调用某个函数。静态多态主要体现为函数重载运算符重载

函数重载即类中定义多个同名成员函数,函数参数类型、参数个数和返回值不完全相同,编译器编译后这些同名函数的函数名会不一样,也就是说编译期间就确定了调用某个函数。C语言函数编译后函数名就是原函数名,C++函数名为原函数名拼接函数参数等信息。

动态多态即运行时多态,在程序执行期间(非编译期)判断所引用对象的实际类型,根据其实际类型调用相应的方法。动态多态由虚函数来实现。
比如


class Base{};
class A: public Base{};
class A: public Base{};

Base *base = new A; // base静态类型为Base*,动态类型为A*
base = new B; // base动态类型变为B*了

探索虚函数表结构

之前的文件提到过,一个类占用的空间,如果有虚函数就会占用8字节的空间来存放虚函数表的地址。
虚函数表内存空间 中依次存放着各个虚函数的指针,通过这个指针可以调用相关的虚函数。

下面通过代码来验证一下上面这个内存结构,定义一个Base类,中间有3个方法,f1/f2/f3。


class Base {
public:
    virtual void f1(){
        std::cout << __PRETTY_FUNCTioN__ << std::endl;
    }

    virtual void f2(){
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f3(){
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

实例化这个类后的内存模型如下图所示:
在这里插入图片描述

下面通过代码来验证这个内存模型。


int main() {
    typedef void(*Fun)();  // Fun为f1 f2 f3的函数类型

    std::cout << sizeof(Base)<< std::endl;  // 输出 8

    Base b;
    printf("b ptr = %p\n", &b);  // b ptr = 0x7ffeee41ac30

    long v_table_addr_value = *(long*)&b; // 取&b指针 前8字节的值,即虚函数表地址值
    printf("vtable ptr = 0x%lx\n", v_table_addr_value); // vtable ptr = 0x557dae962d48

    void *v_table_addr = (void*)v_table_addr_value;  // 把这8字节值转为地址,即为虚函数表指针
    printf("vtable ptr = %p\n", v_table_addr); // vtable ptr = 0x557dae761cd4

    long f1_addr_value = *(long*)v_table_addr;  // 虚函数表前8字节为f1()函数指针值
    printf("f1() ptr = 0x%lx\n", f1_addr_value);  // f1() ptr = 0x557dae761cd4
    Fun f1 = (Fun)f1_addr_value;  // 虚函数表内存第1个8字节值转为函数指针
    f1();  // 输出:virtual void Base::f1()

    long f2_addr_value = *(long*)((char*)v_table_addr + 8);  // 虚函数表8-16字节为f2()函数指针值
    printf("f2() ptr = 0x%lx\n", f2_addr_value);  // f2() ptr = 0x557dae761d0c
    Fun f2 = (Fun)f2_addr_value;  // 虚函数表内存第2个8字节值转为函数指针
    f2();  // 输出:virtual void Base::f2()

    long f3_addr_value = *(long*)((char*)v_table_addr + 16);  // 虚函数表前16-24字节为f3()函数指针值
    printf("f3() ptr = 0x%lx\n", f3_addr_value);  // f3() ptr = 0x557dae761d44
    Fun f3 = (Fun)f3_addr_value;  // 虚函数表内存第3个8字节值转为函数指针
    f3();  // virtual void Base::f3()

    return 0;
}

通过上述代码的输出结果可以验证上图的内存模型。

继承基类重写虚函数

现在定义一个继承类Derived,重写了f1()函数,也就是覆盖掉了Base类中的函数f1()。同时又新增了虚拟函数f4()


class Base {
public:
    virtual void f1(){
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f2(){
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f3(){
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

class Derived : public Base
{
public:
    virtual void f1() override {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f4() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

通过上一节类似的代码可以验证new Derived()其内存模型为
在这里插入图片描述
由此可以得出以下结论:

  1. 虚函数按照其声明顺序放于表中。
  2. 父类的虚函数在子类的虚函数前面。
  3. 覆盖的函数放到了虚函数表中原来父类虚函数的位置。
  4. 没有被覆盖的虚函数函数位置不变。

继承N个基类就有N个虚函数表,接下来使用代码去验证。

有3个基类Base1,Base2, Base3,都有两个虚函数f1()、f2()。最后Derived 类继承这3个基类。并重写f1()函数,新增f4()函数。


class Base1 {
public:
    virtual void f1() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f2() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

class Base2 {
public:
    virtual void f1() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f2() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

class Base3 {
public:
    virtual void f1() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f2() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

class Derived : public Base1, public Base2, public Base3 {
public:
    void f1() override {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }

    virtual void f4() {
        std::cout << __PRETTY_FUNCTION__ << std::endl;
    }
};

此时,sizeof(Derived) 等于24,可以基本确定类实例中有3个虚函数表指针。
下面通过代码来检查一下内存数据。


int main() {
    typedef void(*Fun)();

    std::cout << sizeof(Derived) << std::endl; // 24

    Derived *d = new Derived();
    printf("b ptr = %p\n", d); // b ptr = 0x5624201d9280

    long v_table1_addr_value = *(long *) d;  // 第1个虚函数表地址值
    printf("vtable1 ptr = 0x%lx\n", v_table1_addr_value); // vtable1 ptr = 0x56241e42ac48

    long b1f1_addr_value = *(long *) v_table1_addr_value;
    printf("b1f1() ptr = 0x%lx\n", b1f1_addr_value); // b1f1() ptr = 0x56241e22a170
    Fun b1f1 = (Fun) b1f1_addr_value;
    b1f1(); // virtual void Derived::f1()

    long b1f2_addr_value = *((long *) v_table1_addr_value + 1);
    printf("b1f2() ptr = 0x%lx\n", b1f2_addr_value); // b1f2() ptr = 0x56241e22a058
    Fun b1f2 = (Fun) b1f2_addr_value;
    b1f2(); // virtual void Base1::f2()

    long b1f3_addr_value = *((long *) v_table1_addr_value + 2);
    printf("b1f3() ptr = 0x%lx\n", b1f3_addr_value); // b1f3() ptr = 0x56241e22a1b4
    Fun b1f3 = (Fun) b1f3_addr_value;
    b1f3(); // virtual void Derived::f3()

    long v_table2_addr_value = *((long *) d + 1); // 类实例内存第2个8字节为 第2个虚函数表地址值
    printf("vtable2 ptr = 0x%lx\n", v_table2_addr_value); // vtable2 ptr = 0x56241e42ac70

    long b2f1_addr_value = *(long *) v_table2_addr_value;
    printf("b2f1() ptr = 0x%lx\n", b2f1_addr_value); // b2f1() ptr = 0x56241e22a1ad
    Fun b2f1 = (Fun) b2f1_addr_value;
    b2f1(); // virtual void Derived::f1()

    long b2f2_addr_value = *((long *) v_table2_addr_value + 1);
    printf("b2f2() ptr = 0x%lx\n", b2f2_addr_value); // b2f2() ptr = 0x56241e22a0c8
    Fun b2f2 = (Fun) b2f2_addr_value;
    b2f2(); // virtual void Base2::f2()

    long b2f3_addr_value = *((long *) v_table2_addr_value + 2);
    printf("b2f3() ptr = 0x%lx\n", b2f3_addr_value); // b2f3() ptr = 0xfffffffffffffff0


    long v_table3_addr_value = *((long *) d + 2); // 类实例内存第3个8字节为 第3个虚函数表地址值
    printf("vtable3 ptr = 0x%lx\n", v_table3_addr_value); // vtable3 ptr = 0x56241e42ac90

    long b3f1_addr_value = *(long *) v_table3_addr_value;
    printf("b3f1() ptr = 0x%lx\n", b3f1_addr_value); // b3f1() ptr = 0x56241e22a1a7
    Fun b3f1 = (Fun) b3f1_addr_value;
    b3f1(); // virtual void Derived::f1()

    long b3f2_addr_value = *((long *) v_table3_addr_value + 1);
    printf("b3f2() ptr = 0x%lx\n", b3f2_addr_value); // b3f2() ptr = 0x56241e22a138
    Fun b3f2 = (Fun) b3f2_addr_value;
    b3f2(); // virtual void Base3::f2()

    return 0;
}

根据上述代码输出结果,可以画出下面内存模型。

在这里插入图片描述

由此可以得出以下结论:

  1. 有几个基类就有几个虚函数表,且实例中虚函数表地址值存储顺序就是基类继承顺序。
  2. 继承类新增的虚函数f3()排在第一个虚函数表中,且在基类虚函数后面。
  3. 继承类中重写基类的虚函数f1(),在每个虚函数表中都覆盖相应的虚函数。

寻找被覆盖的虚函数

Derived 类重写基类Base的f1()函数后,那如果想调用基类的被覆盖的虚函数的话,就需要明确类名字调用。


  Derived *d = new Derived();
    d->f1();  // virtual void Derived::f1()
    d->Base::f1();  // virtual void Base::f1()

内存空间中继承类重写的函数存在于虚函数表中原函数的位置,那么原虚函数的位置在哪呢?

到此这篇关于聊一聊C++虚函数表的问题的文章就介绍到这了,更多相关C++虚函数表内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 聊一聊C++虚函数表的问题

本文链接: https://www.lsjlt.com/news/154679.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 聊一聊C++虚函数表的问题
    之前只是看过C++虚函数表相关介绍,今天有空就来写代码研究一下。 面向对象的编程语言有3大特性:封装、继承和多态。C++是面向对象的语言(与C语言主要区别),所以C++也拥有多态的特...
    99+
    2024-04-02
  • 聊聊c++数组名称和sizeof的问题
    一维数组名称的用途: 可以统计整个数组在内存中的长度 可以获取数组在内存中的首地址 示例: int main() { //数组名用途 //1、可以获取整个数组占用内...
    99+
    2024-04-02
  • 一起聊聊C++中的特殊成员函数
    目录一:背景二:特殊成员函数1. 默认构造函数2. 析构函数3. 赋值构造函数4. 赋值运算符一:背景 在C#中要说类默认给我们定义的特殊成员函数,莫过于构造函数,但在 C++ 中这...
    99+
    2024-04-02
  • 聊聊PHP escapeshellarg函数使用的中文问题
    本篇文章给大家带来了关于PHP escapeshellarg函数的相关知识,其中主要介绍了在命令行中调用 escapeshellarg函数有关中文字符的问题,感兴趣的朋友,下面一起来看一下吧,希望对大家有帮助。PHP 中命令行调用 esca...
    99+
    2023-05-14
    escapeshellarg php
  • 一文聊聊golang中的整数相除问题
    Go是一种非常流行的编程语言,适用于编写高度可伸缩的网络服务器和分布式系统。在这篇文章中,我们将讨论Go语言中的整数相除问题。在Go中,整数相除的结果是一个整数,这意味着如果两个整数相除的结果是一个小数,则小数部分将被截断,并且只保留整数部...
    99+
    2023-05-14
  • 聊聊PythonString型列表求最值的问题
    最近在写一个项目的时候发现了个很神奇的问题 temp_list=['9','10','10','9','10'] print(max(temp_list)) # 9 print(mi...
    99+
    2024-04-02
  • 聊聊Navicat统计的行数竟然和表实际行数不一致的问题
    背景 近期为了保障线上数据库的稳定性,我决定针对一些大表的历史数据有计划地进行备份迁移,但是呢,发现一个奇特的现象,Navicat统计行数和表自身count统计数竟然不一致!?0.0...
    99+
    2024-04-02
  • 聊聊React onClick 传递参数的问题
     背景说明 在下图这样的列表中,点击删除按钮需要执行删除操作  列表产生: { title: '操作', dataIndex: 'rowguid'...
    99+
    2024-04-02
  • 简单聊一聊redis过期时间的问题
    目录1.多次修改一个Redis的String过期键,如何保证他仍然能保留第一次设置时的删除时间2.修改hash、set、Zset、list的值,会使过期时间重置吗?总结1.多次修改一个redis的String过期键,如何...
    99+
    2023-04-14
    redis 过期时间 redis 过期时间 java redis设置过期时间
  • 聊聊spring继承的问题
    目录spring继承的问题为什么输出是0呢?spring注入有继承关系的类通过配置文件通过注解新建一个泛型类新建BaseRepository和BaseService的子类在Sprin...
    99+
    2024-04-02
  • 聊聊mybatissql的括号问题
    目录mybatis sql的括号问题mybatis多层括号(超过三层)解析不了mybatis sql的括号问题 因为一段sql  要关联 A,B,C三个表,查三个表里的数据...
    99+
    2024-04-02
  • 聊聊Druid register mbean error的问题
    key: [com.alibaba.druid.stat.DruidDataSourceStatManager.addDataSource(DruidDataSourceStatMa...
    99+
    2024-04-02
  • 聊聊Spring MVC JSON数据交互的问题
    我们在开发中后端经常需要接受来自于前端传递的Json字符串数据,怎么把Json字符串转换为Java对象呢?后端也经常需要给前端返回Json字符串,怎么把Java对象数据转换为Json...
    99+
    2024-04-02
  • 详细聊聊c语言中的缓冲区问题
    目录发现问题例题问题原因解决方法一:解决方法二:解决方案三:出错二gets函数引入为什么要引入缓冲区总结发现问题 你是不是总会出现当你输入的时候(你想的是只输出一个内容),但是最后却...
    99+
    2024-04-02
  • 简单聊聊C++中回调函数的实现
    目录前言1 函数指针2 C风格的回调函数3 C++风格的回调函数4 多态类型的回调函数5 通过function和bind实现回调函数功能6 总结前言 回调函数其实和普通函数一样,不同...
    99+
    2024-04-02
  • 聊聊SpringBoot的@Scheduled的并发问题
    目录SpringBoot @Scheduled的并发spring @Scheduled 并发执行SpringBoot @Scheduled的并发 由于SpringBoot自带的@Sc...
    99+
    2024-04-02
  • 聊聊springboot 整合 hbase的问题
    springboot 整合 hbase 要确定这三个端口外包可以访问 如果是127.0.0.1 可以参考修改 Linux下Hbase安装配置 <property> ...
    99+
    2024-04-02
  • 聊聊.Net,Core配置Nlog.md的问题
    首先在你的项目中用Nuget安装以下两个类库NLog.Extensions.Logging和NLog.Web.AspNetCore然后新建NLog的配置文件Nlog.config,内...
    99+
    2024-04-02
  • 聊聊springboot中整合log4g2的问题
    1.导入jar springboot默认是用logback的日志框架的,所以需要排除logback,不然会出现jar依赖冲突的报错。 <dependency> ...
    99+
    2024-04-02
  • 一文聊聊Go语言中资源竞争问题
    我们都知道,在并发编程中,线程安全是非常重要的。接下来我们就假定一个场景,复现一下线程不安全的情况,再聊聊如何在Go中解决场景我们现在需要对1~100求他们的阶乘,并将结果放到一个map中1! = 1 = 1 2! = 1 * 2 = 2 ...
    99+
    2023-05-14
    Golang go语言 Go 后端
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作