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

C++ 多态以及多态的原理

文章目录

    • 多态的概念
    • 多态的构成条件
      • 虚函数的重写
        • 虚函数重写的两个例外
    • 重载、重写(覆盖)、重定义(隐藏)对比
    • C++11 final 和 override关键字
    • 抽象类
    • 接口继承和普通继承
    • 多态的原理
      • 虚函数表
      • 多态的原理
    • 单继承和多继承关系的虚函数表
      • 单继承中的虚函数表
      • 多继承中的虚函数表

多态的概念

多态的概念:通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会 产生出不同的状态
比如,买票时都是同一个景点有学生票半价和成人票全价等等

多态的构成条件

多态的构成条件主要涉及两个概念:虚函数和继承。
虚函数

  • 虚函数是C++中用于实现运行时多态性的关键概念。
  • 被virtual修饰的类成员函数称为虚函数
class Person {
public://虚函数virtual void BuyTicket() {//....}
};

继承中构成多态还需要满足两个条件

  • 被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写
  • 父类的指针或者引用进行调用

举个栗子

class Person {
public://虚函数virtual void BuyTicket() {cout << "买票-全价" << endl;}
};
class Student : public Person
{//重写基类函数virtual void BuyTicket(){cout << "买票-半价" << endl;}
};
//必须是基类对象指针或引用调用
void Func(Person& people)
{people.BuyTicket();
}
void Test()
{Person Mike;Func(Mike);Student Johnson;Func(Johnson);
}
int main()
{Test();return 0;
}

父类对象和子类对象调用同一个函数,得到的结果不一样

运行结果

image.png
解释

image.png

虚函数的重写

虚函数重写(也叫覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的 返回值类型、函数名字、参数列表完全相同),称派生类的虚函数重写了基类的虚函数。

在上面的例子中 派生类Student中的BuyTicket函数就重写了基类Person的虚函数

虚函数重写的两个例外
  1. 协变
    派生类重写基类虚函数时,与基类虚函数返回值类型不同,且返回值必须是父子关系的指针或者引用。即基类虚函数返回基类对象的指 针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变
    比如
class Person
{
public:// 基类虚函数返回基类指针virtual Person* BuyTicket() {return new Person();}
};
class Student : public Person
{// 派生类协变,返回更具体的类型 Student*virtual Student* BuyTicket(){return new Student();}
};

上面例子中也构成虚函数的重写,派生类和基类的返回值不同,称为协变

  1. 析构函数的重写(基类与派生类析构函数的名字不同)
    如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字, 都与基类的析构函数构成重写,虽然基类与派生类析构函数名字不同。虽然函数名不相同, 看起来违背了重写的规则,其实不然,这里可以理解为编译器对析构函数的名称做了特殊处 理,编译后析构函数的名称统一处理成destructor。
class Person
{
public:virtual ~Person(){cout << "~Person()" << endl;}
};
class Student : public Person
{
public:~Student(){cout << "~Student()" << endl;}
};
int main()
{Person* p1 = new Person();Person* p2 = new Student();// 只有派生类Student的析构函数重写了Person的析构函数,下面的delete对象调用析构函数//才能构成多态,才能保证p1和p2指向的对象正确的调用析构函数。delete p1;delete p2;
}

运行结果:

image.png
12行加不加virtual关键字 都构成重写

重载、重写(覆盖)、重定义(隐藏)对比

image.png

C++11 final 和 override关键字

C++对函数重写的要求比较严格,有些情况可能由于疏忽,导致无法构成重写,这种情况编译器不会报错,程序会正常运行,但是得到的结果不是正确的,所以C++11引入了final和override关键字

  • final 修饰虚函数,表示该虚函数不能再被重写
class Person
{virtual void Func() final{cout << "virtual void Func() final" << endl;}
};class Student : public Person
{virtual void Func()//error{cout << "virtual void Func()" << endl;}
};

编译时报错:

image.png

  • override 检查派生类是否重写了基类虚函数。如果没重写,编译器会报错。
class Person
{
public:virtual void Func() const {cout << "virtual void Func() " << endl;}
};class Student : public Person
{
public:virtual void Func() override//error 派生类没有正确重写基类Func函数  编译器会报错,少了const修饰{cout << "virtual void Func()" << endl;}
};

抽象类

在虚函数的后面写上 =0 ,则这个函数为纯虚函数。包含纯虚函数的类叫做抽象类(也叫接口 类),抽象类不能实例化出对象。派生类继承后也不能实例化出对象,只有重写纯虚函数,派生类才能实例化出对象。纯虚函数规范了派生类必须重写,另外纯虚函数更体现出了接口继承。

class Person
{
public:virtual void Abstract() = 0;//纯虚函数  Person类为抽象类  Person类不能实例化出对象
};//派生类
class Student : public Person
{
public:virtual void Abstract() override//派生类必须重写纯虚函数,派生类才可以实例化出对象,否则不行{cout << "Hello World\n";}};
int main()
{//Person p1; //error 抽象类无法实例化对象Student s1;Person* s1prt = &s1;//使用基类指针访问s1prt->Abstract();return 0;
}

接口继承和普通继承

  • 普通继续 派生类继承了基类,可以使用函数,继承的是函数的实现
  • 接口继承,虚函数的继承是一种接口继承,派生类继承的是基类虚函数的接口,目的是为了重写,达成多态,继承的是接口。所以如果不实现多态,不要把函数定义成虚函数。

多态的原理

虚函数表

#include <iostream>
using namespace std;
class Person
{
public:virtual void Func();
private:int _a;char _b;
};
int main()
{cout << sizeof(Person) << endl;return 0;
}

上面代码求Person所占字节数大小
按照内存对齐的规则,Person类的大小应该是8(32位下)。
但实际结果是12

image.png
这里不仅要内存对齐,当实例化一个对象后发现,成员变量不仅仅只有_a,和_b 。还有一个指针_vfptr(虚函数表指针)

image.png

  • 一个含有虚函数的类都会有至少一个虚函数表指针,虚函数表指针存放在对象的前4个字节或者前8个字节(32位下4个字节,64位下8个字节)。
  • 而虚函数的地址会被存放到虚函数表中。虚函数表也简称为虚表
  • 虚函数表指针指向虚表。
  • 虚函数表本质是一个存虚函数指针的指针数组,一般情况这个数组最后面放了一个nullptr(VS编译器下做了处理,g++没有处理)

image.png
一个含有虚函数的类中都至少都有一个虚函数表指针,因为虚函数的地址要被放到虚函数表中,虚函数表也简称虚表。
通过下面的代码进行分析派生类中的虚表。

class Person
{
public:virtual void Func(){cout << "virtual void Func()" << endl;}virtual void Func2(){cout << "virtual void Func()" << endl;}void Func3()//普通函数{cout << "void Func3()" << endl;}
private:int _a = 0;char _b = 0;
};
class Student : public Person
{virtual void Func() override//重写基类函数{cout << "virtual void Func()" << endl;}
};
int main()
{	Person p1;//基类对象Student s1;//派生类对象return 0;
}
  • Person类中有两个虚函数,一个非虚函数。
  • Student继承了Person类,并且重写了Func函数。
  • 实例化出基类和派生类对象 监视窗口如下
    image.png
    可以发现:
  • 派生类对象也有一个虚表指针,虚表由两部分组成,一部分是继承基类的成员,另一部分是自己的成员。
  • 基类对象和派生类对象的虚表是不一样的,派生类重写了基类的Func虚函数,所以派生类对象虚表中存的是派生类重写后的函数地址。所以虚函数的重写也覆盖,覆盖值得是虚表中虚函数的覆盖,重写是语法的叫法,覆盖是底层的原理。
  • Func2虚函数被继承下来后也会被放到虚表中,Func3也会被继承下来,但是Func3不是虚函数,所以不会放到虚表中

多态的原理

class Person {
public://虚函数virtual void BuyTicket(){cout << "买票-全价" << endl;}
};
class Student : public Person
{//重写基类函数virtual void BuyTicket(){cout << "买票-半价" << endl;}
};
//必须是基类对象指针或引用调用
void Func(Person& people)
{people.BuyTicket();
}
void Test()
{Person p1;Func(p1);Student s1;Func(s1);
}

对于上面的例子

image.png

  • 当people指向的是基类对象时,people.BuyTicket();就会在基类对象p1中的虚表中找到对应虚函数
  • 当people指向的是派生类对象时,,people.BuyTicket();就会在派生类对象s1中的虚表中找到对应的虚函数
  • 通过虚表,实现了了不同对象去完成同一行为时,展现出不同的形态。

单继承和多继承关系的虚函数表

单继承中的虚函数表

class Base 
{
public:virtual void func1() { cout << "Base::func1" << endl; }virtual void func2() { cout << "Base::func2" << endl; }
private:int a;
};
class Derive :public Base 
{
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }virtual void func4() { cout << "Derive::func4" << endl; }
private:int _b;
};
int main()
{Base b1;Derive d1;return 0;
}
  • 上面代码中基类Base有两个虚函数
  • 派生类Derive继承了Base类,并且重写了func1函数,且新增了两个虚函数func3 和func4

单继承对象模型
image.png

image.png

通过监视窗口发现:派生类中新增的虚函数func3 和func4 没有进虚函数表。(不知道是编译器故意的 还是编译器的
bug)
image.png
我们可以利用程序自己打印虚表来观察,参考代码如下。

class Base 
{
public:virtual void func1() { cout << "Base::func1" << endl; }virtual void func2() { cout << "Base::func2" << endl; }
private:int a;
};
class Derive :public Base 
{
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }virtual void func4() { cout << "Derive::func4" << endl; }
private:int b;
};
typedef void (*VF_Ptr)();//函数指针
//VF_Prt table[];//函数指针数组//打印虚函数表
void PrintVFTable(VF_Ptr table[])
{for (int i = 0; table[i] != nullptr; ++i){printf("table[%d] = %p\n", i, table[i]);VF_Ptr Fun = table[i];//取出函数地址对其进行访问Fun();}cout << endl;
}
int main()
{Base b1;Derive d1;//虚函数表指针在对象的头四个字节(32位下), 拿到对象的地址对其强制类型转换:(int*)&p1//在解引用就能拿到对象前四个字节地址:*((int*)&p1),在将其强制类型转换位函数指针:(VF_Ptr*)(*(int*)&p1)PrintVFTable((VF_Ptr*)(*(int*)&b1));PrintVFTable((VF_Ptr*)(*(int*)&d1));return 0;
}

运行结果:

image.png
可以看出,不论是派生类还是基类,只要是虚函数都会存到虚表中

多继承中的虚函数表

class Base1 {
public:virtual void func1() { cout << "Base1::func1" << endl; }virtual void func2() { cout << "Base1::func2" << endl; }
private:int _a;
};
class Base2 {
public:virtual void func1() { cout << "Base2::func1" << endl; }virtual void func2() { cout << "Base2::func2" << endl; }
private:int _b;
};
class Derive : public Base1, public Base2 {
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }
private:int _d;
};

多继承对象模型
多继承对象模型对比单继承模型就复杂很多

image.png

image.png

image.png
派生类会有两个虚表,监视窗口仍然无法观察, 通过程序打印查看

typedef void(*VFPTR) ();
void PrintVTable(VFPTR vTable[])
{cout << " 虚表地址>" << vTable << endl;for (int i = 0; vTable[i] != nullptr; ++i){printf(" 第%d个虚函数地址 :0X%x,->", i, vTable[i]);VFPTR f = vTable[i];f();}cout << endl;
}
int main()
{Derive d;VFPTR* vTableb1 = (VFPTR*)(*(int*)&d);PrintVTable(vTableb1);//派生类第二个虚表指针需要加行Base对象大小的偏移量才能获得VFPTR* vTableb2 = (VFPTR*)(*(int*)((char*)&d + sizeof(Base1)));PrintVTable(vTableb2);return 0;
}

结果如下

image.png
和上面的对象模型一样。
可以发现:
多继承派生类的未重写的虚函数放在第一个继承基类部分的虚函数表中

相关文章:

C++ 多态以及多态的原理

文章目录 多态的概念多态的构成条件虚函数的重写虚函数重写的两个例外 重载、重写(覆盖)、重定义(隐藏)对比C11 final 和 override关键字抽象类接口继承和普通继承多态的原理虚函数表多态的原理 单继承和多继承关系的虚函数表单继承中的虚函数表多继承中的虚函数表 多态的概念 …...

贝蒂详解<string.h>(下)

✨✨欢迎大家来到贝蒂大讲堂✨✨ ​​​​&#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C语言学习 贝蒂的主页&#xff1a;Betty‘s blog 目录 1. 简介 2. memset()函数 2.1用法 2.2实例 2.3 实现me…...

问题 F: 分巧克力

题目描述 儿童节那天有 K 位小朋友到小明家做客。小明拿出了珍藏的巧克力招待小朋友们。小明一共有 N 块巧克力&#xff0c;其中第i 块HiWi 的方格组成的长方形。 为了公平起见&#xff0c;小明需要从这 N 块巧克力中切出 K 块巧克力分给小朋友们。 切出的巧克力需要满足&am…...

安装pillow可能遇到的问题

安装命令 pip install Pillow安装 Pillow 这个 Python 图像处理库时可能会遇到多种问题。以下一些常见的安装问题及其解决方法&#xff1a; 缺少依赖项: Pillow 安装可能需要一些基础库&#xff0c;如 libjpeg 和 zlib。如果在安装时提示缺少这些库&#xff0c;你需要先安装它…...

详解ajax、fetch、axios的区别

众所周知它们都用来发送请求&#xff0c;其实它们区别还蛮大的。这也是面试中的高频题&#xff0c;本文将详细进行讲解。 1. ajax 英译过来是Aysnchronous JavaScript And XML&#xff0c;直译是异步JS和XML&#xff08;XML类似HTML&#xff0c;但是设计宗旨就为了传输数据&a…...

致远OA getAjaxDataServlet XXE漏洞复现(QVD-2023-30027)

0x01 产品简介 致远互联-OA 是数字化构建企业数字化协同运营中台,面向企业各种业务场景提供一站式大数据分析解决方案的协同办公软件。 0x02 漏洞概述 致远互联-OA getAjaxDataServlet 接口处存在XML实体注入漏洞,未经身份认证的攻击者可以利用此漏洞读取系统内部敏感文件…...

力扣最热一百题——只出现一次的数字

这个合集已经很久没有更新了&#xff0c;今天来更新更新~~~ 目录 力扣题号 题目 题目描述 示例 提示 题解 Java解法一&#xff1a;Map集合 Java解法二&#xff1a;位运算 C位运算代码 力扣题号 136. 只出现一次的数字 - 力扣&#xff08;LeetCode&#xff09; 下述题…...

UE5 UE4 修复GPU驱动程序崩溃

原贴链接&#xff1a;https://mp.weixin.qq.com/s/e5l9XtfwEFWgwhHi1b2idg UE5 UE4在处理含有大量图形的项目时&#xff0c;你有可能会遇到GPU崩溃 可以通过修改注册表&#xff0c;修复崩溃。 GPU崩溃情况概述 UE5 UE4在处理含有大量图形的项目时&#xff0c;你有可能会遇到G…...

SpiderFlow爬虫平台 前台RCE漏洞复现(CVE-2024-0195)

0x01 产品简介 SpiderFlow是新一代爬虫平台,以图形化方式定义爬虫流程,以流程图的方式定义爬虫,不写代码即可完成爬虫,是一个高度灵活可配置的爬虫平台。 0x02 漏洞概述 SpiderFlow爬虫平台src/main/java/org/spiderflow/controller/FunctionController.java文件的Functi…...

帆软report 设置条件属性,值为负数标为红色功能时,不生效

详细情况&#xff1a; 在设置负数为红色功能前&#xff0c;已经有一个条件属性&#xff0c;数据集获取的值为空或者为0时&#xff0c;转换成 - 符号。如下图&#xff1a; 具体表单显示效果如下&#xff1a; 条件属性2设置 原因 因为条件属性1设置的 - 符号没有设置颜色&#xf…...

QML实现的图片浏览器

很久之前实现了一个QWidget版本的图片浏览器:基于Qt5的图片浏览器QHImageViewer 今天用QML也实现一个,功能差不多: ●悬浮工具栏 ●支持图片缩放、旋转、还原、旋转、拖动。 ●拖动图片时,释放鼠标图片会惯性滑动。 ●支持左右翻页查看文件夹中的图片。 ●支持保存图片至本…...

【HTML】对字体的所有操作详解(经典)

目录 一、文字样式设置的基本标签二 、 设置文字的颜色三、设置文字的尺寸四、 设置文字的字体五、 使文字倾斜六、 使文字加粗七、处理网页中的特殊字符十、 如何更方便地忽略浏览器对部分HTML的解析十一、 其他文字修饰方法十二、为了让文字富有变化&#xff0c;或者为了着意…...

关于调查项目的讨论

怎么安排一个调查项目 要安排一个调查项目&#xff0c;你需要经过以下步骤&#xff1a; 1. 确定调查目的&#xff1a;明确你为什么要进行这个调查&#xff0c;你想了解什么问题或获得什么信息。 2. 制定研究问题&#xff1a;根据调查目的&#xff0c;确定需要回答的具体问题…...

Matlab三维绘图

绘制三维图plot3 t0:pi/50:10*pi; xsin(t); ycos(t); zt; plot3(x,y,z); 产生栅格数据点meshgrid 这个接口在绘制三维图像里面相当重要&#xff0c;很多时候要将向量变成矩阵才能绘制三维图。 x0:0.5:5; y0:1:10; [X,Y]meshgrid(x,y); plot(X,Y,o); x和y是向量&#xff0c;…...

一体式气象站的优点是什么?带大家了解一下

一体式气象站是一款高度集成、低功耗、可快速安装、便于野外监测使用的高精度自动气象观测设备。 一体式气象站的优点主要体现在以下几个方面&#xff1a; 集成度高&#xff1a;一体式气象站集成了多种气象传感器、数据处理单元、显示单元和通讯模块等&#xff0c;可以同时监…...

第八讲_css定位

css定位 1. css定位介绍2. 静态定位&#xff08;static&#xff09;3. 相对定位&#xff08;relative&#xff09;4. 绝对定位&#xff08;absolute&#xff09;5. 固定定位&#xff08;fixed&#xff09;6. 粘性定位&#xff08;sticky&#xff09; 1. css定位介绍 在 css 中…...

找出字符串中第一个匹配项的下标(Leetcode28)

例题&#xff1a; 分析&#xff1a; 题目的意思就是&#xff1a; 先给出一个字符串pattern&#xff0c;要拿着pattern字符串和原始字符串&#xff08;origin&#xff09;比对&#xff0c;若在origin中找到了pattern字符串&#xff0c;则返回pattern字符串在原始字符串origin中的…...

【分布式微服务专题】从单体到分布式(四、SpringCloud整合Sentinel)

目录 前言阅读对象阅读导航前置知识一、什么是服务雪崩1.1 基本介绍1.2 解决方案 二、什么是Sentinel2.1 基本介绍2.2 设计目的2.3 基本概念 三、Sentinel 功能和设计理念3.1 流量控制3.2 熔断降级3.3 系统负载保护 四、Sentinel 是如何工作的 笔记正文一、简单整合Sentinel1.1…...

RHCE9学习指南 第19章 网络时间服务器

19.1 时间同步的必要性 对于一些服务来说对时间要求非常严格&#xff0c;例如&#xff0c;图19-1所示由三台服务器搭建的ceph集群。 图19-1 三台机器搭建的集群对时间要求比较高 这三台服务器的时间必须要保持一样&#xff0c;如果不一样&#xff0c;就会显示报警信息。那么…...

大模型 RAG 问答技术架构及核心模块盘点:从 Embedding、prompt-embedding 到 Reranker

对于RAG而言&#xff0c;2023年已经出现了很多工作&#xff0c;草台班子有了一堆&#xff0c;架构也初步走通&#xff0c;2024年应该会围绕搜索增强做更多的优化工作。 因此我们今天来系统回顾下RAG中的模块&#xff0c;包括一些架构&#xff0c;文本嵌入embedding等&#xff…...

在rocky linux 9.5上在线安装 docker

前面是指南&#xff0c;后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...

Leetcode 3577. Count the Number of Computer Unlocking Permutations

Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接&#xff1a;3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯&#xff0c;要想要能够将所有的电脑解锁&#x…...

css3笔记 (1) 自用

outline: none 用于移除元素获得焦点时默认的轮廓线 broder:0 用于移除边框 font-size&#xff1a;0 用于设置字体不显示 list-style: none 消除<li> 标签默认样式 margin: xx auto 版心居中 width:100% 通栏 vertical-align 作用于行内元素 / 表格单元格&#xff…...

scikit-learn机器学习

# 同时添加如下代码, 这样每次环境(kernel)启动的时候只要运行下方代码即可: # Also add the following code, # so that every time the environment (kernel) starts, # just run the following code: import sys sys.path.append(/home/aistudio/external-libraries)机…...

快刀集(1): 一刀斩断视频片头广告

一刀流&#xff1a;用一个简单脚本&#xff0c;秒杀视频片头广告&#xff0c;还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农&#xff0c;平时写代码之余看看电影、补补片&#xff0c;是再正常不过的事。 电影嘛&#xff0c;要沉浸&#xff0c;…...

6️⃣Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙

Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙 一、前言:离区块链还有多远? 区块链听起来可能遥不可及,似乎是只有密码学专家和资深工程师才能涉足的领域。但事实上,构建一个区块链的核心并不复杂,尤其当你已经掌握了一门系统编程语言,比如 Go。 要真正理解区…...

虚拟机网络不通的问题(这里以win10的问题为主,模式NAT)

当我们网关配置好了&#xff0c;DNS也配置好了&#xff0c;最后在虚拟机里还是无法访问百度的网址。 第一种情况&#xff1a; 我们先考虑一下&#xff0c;网关的IP是否和虚拟机编辑器里的IP一样不&#xff0c;如果不一样需要更改一下&#xff0c;因为我们访问百度需要从物理机…...

JS的传统写法 vs 简写形式

一、条件判断与逻辑操作 三元运算符简化条件判断 // 传统写法 let result; if (someCondition) {result yes; } else {result no; }// 简写方式 const result someCondition ? yes : no;短路求值 // 传统写法 if (condition) {doSomething(); }// 简写方式 condition &…...

暴雨新专利解决服务器噪音与性能悖论

6月1日&#xff0c;我国首部数据中心绿色化评价方面国家标准《绿色数据中心评价》正式实施&#xff0c;为我国数据中心的绿色低碳建设提供了明确指引。《评价》首次将噪音控制纳入国家级绿色评价体系&#xff0c;要求从设计隔声结构到运维定期监测实现闭环管控&#xff0c;加速…...

2025年ESWA SCI1区TOP,自适应学习粒子群算法AEPSO+动态周期调节灰色模型,深度解析+性能实测

目录 1.摘要2.粒子群算法PSO原理3.改进策略4.结果展示5.参考文献6.代码获取7.算法辅导应用定制读者交流 1.摘要 能源数据的科学预测对于能源行业决策和国家经济发展具有重要意义&#xff0c;尤其是短期能源预测&#xff0c;其精度直接影响经济运行效率。为了更好地提高预测模型…...