当前位置: 首页 > backend >正文

【三大特性】虚表 内存分布

一、虚函数表(vptr)

之前研究到,vptr存储在对象实例的最前面位置
这样意味着我们可以:

  • 对象实例 -> vptr指针 -> 虚函数表位置 -> 各类虚函数

1.1、示例代码

#include "pch.h"
#include <iostream>
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()
{typedef void(*Fun)(void);Base b;Fun pFun = NULL;Base * p = &b;cout << "该对象的地址:" << p << endl;cout << "虚函数表的指针也是从这个地址"<< (int*)(&b) <<"开始存的" << endl << endl;cout << "虚函数表的指针指向的地址10进制:" << *(int*)(&b) << "即虚函数表的指针存的内容"<<endl;cout << "即虚函数表的地址:" << (int*)*(int*)(&b) << endl << endl;pFun = (Fun)*(int*)*(int*)(&b);//第一个虚函数的指针cout << "第一个虚函数的地址:" << pFun << endl;pFun();Fun gFun = NULL;gFun = (Fun)*((int*)*(int*)(&b) + 1);//第二个虚函数的指针Fun hFun = NULL;hFun = (Fun)*((int*)*(int*)(&b) + 2);//第三个虚函数的指针
}

如下所示:

  • b为BaseTest类型对象,&b返回BaseTest类型对象指针
  • (int*)(&b) :把BaseTest类型指针改为int类型,指针指向地址没变,但指向对象的类型变了。
  • *(int*)(&b) : 对int*类型指针解引用,从b对象地址开始,取出sizeof(int)个字节(因为此时指针认为自己指向一个int对象)

前面有做过实验,*(&b)的话取回的还是BaseTest类型,若是改为int*,则可以正常打印b对象地址。

  • (int *)*(int *)(&b) :相当于 *(int *)(&b) 这个int数值前加上(int*) ,即返回一个int指针,此指针代表 b对象首地址的vptr里面的内容,即指向虚函数表。
  • *(int *)*(int *)(&b) :同上面一样,对int*解引用,返回int数值。
  • (Func)*(int *)*(int *)(&b) : Func是函数指针,后接int数值,则代表此函数指针指向这个int数值。
    因此上面6步结束后,得到虚函数表中的虚函数地址,则可以直接调用。
  • (Func)*((int *)*(int *)(&b) + 1): 这里就是数组的指针的正常操作,现在这个指针指向了数组的第二个元素(即第二个虚函数指针),最后就是解引用,然后转换为Fun函数指针。

【注】:这里存在一个问题,VS可得到目标虚函数,但QT得到的虚函数地址只有第一个函数可用?- 参考:https://www.cnblogs.com/cmt/p/14580194.html?

二、虚函数表结束标志

虚函数表有开始就有结束,其的结束标志是 ‘\0’

char* end = NULL;
end = (char*)((int*)*(int*)(&b) + 3);

加上如上代码,这里指向了虚函数表即指针数组的第四个元素,但实际上数组里只有三个指针,所以这里便刚好指向了结束标志。再通过(char*)转换指针类型,代表指向的是一个字节。

【注】:char型存储的含义:

char end1 = '\0';	//字符串的结束符
char end2 = 0;		//字符串的结束符
char zero1 = '0';	//这才是真正的字符0
char zero2 = 48;

可以通过数组应用,来进行循环索引。

三、派生类虚函数表 结构顺序

3.1、单继承(无虚函数覆盖)

例子:基类有三个虚函数,派生类有三个虚函数,但派生类没有另外重写基类虚函数。

虚函数表顺序: 基类虚函数1.2.3 -> 派生类虚函数1.2.3 -> ‘\0’
在这里插入图片描述

3.2、单继承(有虚函数覆盖)

例子:基类有三个虚函数,派生类有三个虚函数,派生类重写基类虚函数f。

虚函数表顺序: 派生类1 -> 基类虚函数.2.3 -> 派生类虚函数2.3 -> ‘\0’
在这里插入图片描述

3.3、多继承(无虚函数覆盖)

例子:有三个基类,一个派生类,且派生类一个虚函数也没有去重写。

虚函数表顺序:

  • 对于继承到的每个基类,都有一个对应的虚函数表。
  • 对于继承到的每个基类,都有一个对应的虚函数表
    派生类的虚函数的指针,被放进了第一个基类对应的虚函数表里。(按照声 明顺序来判断的)
    在这里插入图片描述在这里插入图片描述

3.4、多继承(有虚函数覆盖)

例子:有三个基类,一个派生类,且派生类重写了三个基类的同一个虚函数。

虚函数表顺序:

  • 三个基类的虚函数表的第一项,都被替换为Derive::f的指针
  • 这样任意基类指针指向派生类对象,都可以调用到Derive::f
    在这里插入图片描述
http://www.xdnf.cn/news/2523.html

相关文章:

  • Marmoset Toolbag 5.0 中文汉化版 八猴软件中文汉化版 免费下载
  • C# 类(Class)教程
  • 浮点数:IEEE 754标准
  • PCIe 转 U.2 接双硬盘指南 - 超微(Supermicro)主板
  • Mysql如何高效的查询数据是否存在
  • 理解 Kubernetes 初始访问向量(一)——控制平面
  • 【Webpack \ Vite】多环境配置
  • makefile总结
  • 关于Spark知识点与代码测试的学习总结
  • 单片机 + 图像处理芯片 + TFT彩屏 复选框控件
  • 30-算法打卡-字符串-重复的子字符串-leetcode(459)-第三十天
  • 使用 Cherry Studio 调用高德 MCP 服务
  • NFS从零部署
  • 华为 CCE 查看节点剩余可调度cpu核数
  • 从零实现分布式WebSocket组件:设计模式深度实践指南
  • 路由协议基础
  • babel和loader的关系
  • 微深节能 平板小车运动监测与控制系统 格雷母线
  • LeetCode题解1297. 子串的最大出现次数
  • 低压电工常见知识点
  • 麒麟系统通过 Service 启动 JAR 包的完整指南
  • 【KWDB 创作者计划】_KWDB引领数据库技术革新的璀璨之星
  • 业务校验工具包-validate-utils介绍
  • spring-rabbit的CachingConnectionFactory默认参数导致消费者Channel数量暴增问题解决
  • go语言八股文(三)
  • Deep Dark Sea 局域網文件共享即時匿名聊天去數據庫部署
  • Ldap高效数据同步- MirrorMode双主复制模式配置详解(下)
  • spring项目rabbitmq es项目启动命令
  • 【Pandas】pandas DataFrame rfloordiv
  • 查回来的数据除了 id,其他字段都是 null