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

C++中多态的原理【精华】

虚函数表

通过一道题我们先感受一下编译器针对多态的处理

#include <iostream>
using namespace std;class Base
{
public:virtual void Func1(){cout << "Func1()" << endl;}
private:int _b = 1;char _c
};int main()
{cout << sizeof(Base) << endl;return 0;
}

在x64环境下,显示的结果是12字节
在这里插入图片描述

为什么是12字节呢?一个int就4字节,一个char就1字节,内存对齐一下也是8字节呀!为什么显示的结果是12字节呢? 我们调试代码,看看内部结构
在这里插入图片描述
通过调试我们发现Base创建的对象里面有一个指针_vfptr,如果我们计算sizeof的时候把这个指针算上的话刚刚好的12字节!

那么疑问来了,这个指针到底是什么?为什么会出现在对象里面?这个指针到底有什么作用?

这个指针我们叫做虚函数表指针(v代表virtual,f代表function)。一个含有虚函数的类中都至少都有一个虚函数表指针,因为虚函数的地址要被放到虚函数表中,虚函数表也简称虚表,。那么子类中这个表放了些什么呢?我们接着往下分析

#include <iostream>
using namespace std;class Base
{
public:virtual void Func1(){cout << "Func1()" << endl;}virtual void Func2(){cout << "Func2()" << endl;}void Func3(){cout << "Func3()" << endl;}
private:int _b = 1;char _c;
};int main()
{cout << sizeof(Base) << endl;Base b;return 0;
}

通过监视窗口再来看:
在这里插入图片描述
我们发现这个指针指向一个数组(虚函数表),这个数组(表)里面存的就是虚函数的地址


知道了包含虚函数的类里面会出现一个指针,这个指针指向一个指针数组,数组里面保存的是类里面的虚函数,接下来我们看看多态的原理

文章目录

  • 虚函数表
  • 多态的原理


多态的原理

提示:这里可以添加本文要记录的大概内容:

class Person {
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }virtual void func(){}
private:int _a = 1; 
};class Student : public Person {
public:virtual void BuyTicket() { cout << "买票-半价" << endl; }
private:int _b = 0; 
};void Func(Person& p)
{p.BuyTicket();
}int main()
{Person p;Student s;Func(p);Func(s);return 0;
}

我们看一下父类的虚表指针
在这里插入图片描述
我们看到Person对象里面的_vfptr指向的指针数组里面有两个指针,说明保存的是两个地址,我们看看这两个地址是什么?
在这里插入图片描述
然后我们看一下_vfptr指针指向的数组里面保存的地址
在这里插入图片描述
再看一下子类的虚表指针
在这里插入图片描述
虚函数需要被重写,一开始子类会将虚函数进行直接拷贝,重写虚函数之后,虚函数的地址就发生改变了

那为什么多态的原理和指向的对象有关?和虚表有什么关系?

在这里插入图片描述

父类对象找到虚函数表的地址,在表里面找到虚函数的地址,因为这里是父类对象,直接调用父类对象的虚函数
子类对象赋值给父类对象,这样会产生“切片”,找到子类中父类的那一部分,寻找到虚函数表的地址,但是虚函数表里存的是子类重写的父类虚函数的地址,所以此时_vfptr指向的其实是子类的虚函数指针数组,自认而然掉用的是子类的虚函数

那为什么不使用指针和引用后,就不构成多态了呢?

#include <iostream>
#include <string>
#include <vector>
using namespace std;class Person {
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }virtual void func() {}
private:int _a = 1;
};class Student : public Person {
public:virtual void BuyTicket() { cout << "买票-半价" << endl; }
private:int _b = 0;
};void Func(Person p)
{p.BuyTicket();
}int main()
{Person p;Student s;Func(p);Func(s);return 0;
}

在这里插入图片描述

因为指针和引用是指向子类对象切割出来的父类对象的那一部分
不使用指针和引用后,直接将子类传递给父类,切割出来的子类对象中父类的那一部分,成员拷贝给父类,但是不会拷贝虚表指针

http://www.lryc.cn/news/199034.html

相关文章:

  • 亿赛通电子文档安全管理系统 Update.jsp SQL注入
  • 神经网络中的反向传播:综合指南
  • 协同创新、奔赴未来——“华为云杯”2023人工智能创新应用大赛华丽谢幕
  • 介绍Node.js中fs模块 代码和注释。
  • 【QT 读取JSON】 深入浅出 使用QT内置的QJson模块解析Json文件 匠心之作
  • 初识javaweb2 tomcat
  • 使用OPENROWSET :在一个数据库中查询另一个数据库的数据
  • 基于STM32设计的智慧农业管理系统(ESP8266+腾讯云微信小程序)
  • Flutter视图原理之三棵树的建立过程
  • 详细解析冒泡排序,JS如何基本实现的。
  • 如何消除CSDN博文代码中自动添加的行号
  • 定制效果在线定制印刷系统源码 DIY在线定制系统源码 云印刷定制系统源码手机、PC端实时互通
  • 算法|每日一题|同积元组|哈希统计
  • 最新AI创作系统ChatGPT网站H5源码V2.6.4+搭建部署教程+支持GPT4.0+支持ai绘画(Midjourney)/支持Prompt预设应用
  • 最新!两步 永久禁止谷歌浏览器 Google Chrome 自动更新
  • 在Java中线程和进程的区别
  • 【高危安全通告】Oracle 10月月度安全漏洞预警
  • 卷王问卷考试系统SurveyKing,开源调查问卷和考试系统源码
  • uniapp开发微信小程序,webview内嵌h5,h5打开pdf地址,解决方案
  • Swift使用Embassy库进行数据采集:热点新闻自动生成器
  • 【AIGC核心技术剖析】改进视频修复的传播和变压器(动态滤除环境中的物体)
  • Win系统VMware虚拟机安装配置(二)
  • 基于枚举实现的观察者模式
  • 基于神经网络的图像识别研究
  • 基于SSM的工资管理系统
  • 微服务负载均衡实践
  • php定时任务
  • 2.2 如何使用FlinkSQL读取写入到文件系统(HDFS\Local\Hive)
  • call函数和apply函数的区别
  • JavaCV踩坑之路1——Mac上安装OpenCV