【C++进阶】C++多态概念详解

C++多态概念详解

  • 一,多态概念
  • 二,多态的定义
    • 2.1 多态构成的条件
    • 2.2 什么是虚函数
    • 2.3 虚函数的重写
      • 2.3.1 虚函数重写的特例
      • 2.3.2 override和final
    • 2.4 重载和重写(覆盖)和重定义(隐藏)的区别
  • 三,抽象类
    • 3.1 概念
    • 3.2 接口继承和实现继承
  • 四,多态的原理
    • 4.1 虚函数表
    • 4.2 多态调用的底层原理
    • 4.3 静态绑定和动态绑定
  • 五,单继承和多继承的虚函数表
    • 5.1 单继承的虚函数表
    • 5.2 多继承的虚函数表
  • 六,继承和多态的常见问题

一,多态概念

上节我们看了继承,现在我们来看多态。

那么什么是多态呢?通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态。
举个例子,对于买票这件事,一个成人去买的话是全票,但如果是学生则半价,在这件事中成人和学生都可以买票,但是不同的人买,票价却不同,这就是一种多态行为。

二,多态的定义

2.1 多态构成的条件

多态是在不同继承关系的类对象,去调用同一函数,产生了不同的行为。

在继承中构成多态要满足两个条件:

  1. 在子类中对父类的虚函数进行重写,且必须调用。
  2. 通过父类的指针或者引用调用虚函数

那什么是虚函数及什么是重写,我们下面就来讲解

2.2 什么是虚函数

其实虚函数就是加上virtual的函数, 比如下面的代码:

class Person {
public:virtual void BuyTicket(){cout << "买票-全价" << endl;}
};

2.3 虚函数的重写

虚函数要完成重写,那么重写就是子类中有一个和父类一样的虚函数,这个虚函数要求:

函数名,返回值类型,参数列表相同

class Person {
public:virtual void BuyTicket() {cout << "买票-全价" << endl; }
};
class Student : public Person {
public:virtual void BuyTicket() {cout << "买票-半价" << endl; }
};

2.3.1 虚函数重写的特例

虚函数的重写有两个特例:

  1. 协变----->重写的虚函数的类型可以不一样,但是要是父子类关系的指针或者引用
class A{};
class B : public A {};
class Person {
public:virtual A* f() {return new A;}
};
class Student : public Person {
public:virtual B* f() {return new B;}
};

B这个类是A类的子类,Student类是Person的子类,且都有虚函数f(),但是这两个虚函数的类型分别是A,B父子类的指针,这就是协变


  1. 析构函数的重写 ----->父子类的析构函数会被统一成destuctor,如果不加virtual构成重写,则会构成隐藏,不会调用到父类的析构函数,进而造成内存泄漏(子类的资源没有释放完)
class Person {
public:virtual ~Person() {cout << "~Person()" << endl;}
};
class Student : public Person {
public:virtual ~Student() {cout << "~Student()" << endl;}
};
// 只有派生类Student的析构函数重写了Person的析构函数,下面的delete对象调用析构函
//数,才能构成多态,才能保证p1和p2指向的对象正确的调用析构函数。
int main()
{Person* p1 = new Person;Person* p2 = new Student;delete p1;delete p2;return 0;
}

  1. 虚函数重写时,父类加了virtual,而子类不加virtual也构成重写(建议加上)
class Person {
public:virtual void BuyTicket() {cout << "买票-全价" << endl; }
};
class Student : public Person {
public:void BuyTicket() {cout << "买票-半价" << endl; }
};

2.3.2 override和final

C++中对于重写的要求比较严格,所以有了这两个关键字来检测是否重写


现在有这样一个问题:如何实现一个类,让其不能被继承
有两种办法:

  1. 让父类的构造函数私有,以为子类的构造要用到父类的构造,但是这样会让子类不能实例化出对象
  2. final修饰为最终类

final也可以修饰虚函数,修饰后不能被重写!


override加在派生类后面检查是否完成重写

2.4 重载和重写(覆盖)和重定义(隐藏)的区别

重载我们在前面学过,重写在原理层面也叫覆盖,上一节讲的隐藏也叫重定义。

看下面的图我们可以看到三者的区别:
在这里插入图片描述
其实更深层次来看重写就是一种特殊的重定义!

三,抽象类

3.1 概念

我们先来看什么是纯虚函数,就是在虚函数后面加上 = 0 ,

virtual void fun () = 0

包含纯虚函数的类叫抽象类(接口类),并且抽象类不能实例化对象。

抽象类就像某类事物抽象出来的一个特征,不是一个具体的东西。例如车是一个抽象类,但是像宝马,奥迪,奔驰是车这个抽象类继承的具体的可实例化的类。

抽象类的派生类必须重写虚函数,否则不能实例化,因为不重写子类仍然时抽象类,(间接强制子类重写虚函数)

3.2 接口继承和实现继承

普通函数的继承是一种实现继承,派生类继承基类函数,继承了实现为了复用

虚函数的继承是一种接口继承继承了父类的接口为了重写实现,达成多态

四,多态的原理

普通函数和虚函数都是存在代码段的,谈到多态的原理我们就不得不说下类对象的存储设计
如下图:在这里插入图片描述
一个类中存放着一个指向类成员函数表的指针,而这个表中存放的是函数的地址,多态的原理就和这种存储结构息息相关。

4.1 虚函数表

先来试想一下如何计算一个有虚函数的类的大小:

class Base
{
public:virtual void Func1(){cout << "Func1()" << endl;}
private:int _b = 1;
};
int main() {Base b;cout << sizeof(b) << endl;return 0;
}

运行后我们可以发现
在这里插入图片描述
这是为什么呢?

这是因为Base这个类中除了_b这个成员外,还有一个指针_vfptr,这个指针是虚函数表指针(虚表指针),指向的是虚函数指针数组。
在这里插入图片描述
那么这个指针指向的表是干嘛的呢,我们继续来分析,我们让派生类Derive去继承Base类,并且增加虚函数。

class Base
{
public:virtual void Func1(){cout << "Base::Func1()" << endl;}virtual void Func2(){cout << "Base::Func2()" << endl;}void Func3(){cout << "Base::Func3()" << endl;}
private:int _b = 1;
};class Derive : public Base
{
public:virtual void Func1(){cout << "Derive::Func1()" << endl;}
private:int _d = 2;
};
int main()
{Base b;Derive d;return 0;
}

经过调试我们可以看到
在这里插入图片描述

在Base和Derive类中都有_vfptr指针,指向了一张表,里面貌似存放了虚函数。而且Derive的这个表里第一个存放的是重写的虚函数,第二个存放的是Base的第二个虚函数。

其实这个表是虚函数表(virtual function table),虚表中存储的是虚函数的地址(指针)。
派生类的虚函数表继承自父类的虚函数表,但是会用其自己的虚函数覆盖虚表中第一个位置(所以虚函数的重写也叫覆盖)

重写时语法层面的,覆盖是原理层面的

在这里插入图片描述
虚表以空结尾,并且虚函数存放的顺序和声明的顺序一致

派生类有两部分,一部分是父类的,一部分是自己的,派生类没有自己单独的虚表,而是继承的父类的,拷贝父类的虚函数表,并覆盖自己重写的虚函数


知道了虚表的存在后,我们继续探索。

如果派生类有一个自己的虚函数呢 ? 会在虚表里怎么存放

虚函数表是存放在常量区的,在编译时生成好的,虚表指针的初始化是在构造函数初始化列表最前面(所有对象初始化之前)。

同类型的对象共享一个虚函数表

在这里插入图片描述

4.2 多态调用的底层原理

如何做到指向父类调用父类虚函数,指向子类调子类呢?

class Person {
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }
};
class Student : public Person {
public:virtual void BuyTicket() { cout << "买票-半价" << endl; }
};
void Func(Person& p)
{p.BuyTicket();
}
int main()
{Person Mike;Func(Mike);Student Johnson;Func(Johnson);return 0;
}

运行后可以看到:

在这里插入图片描述
在这里插入图片描述
由上面的图可知,指向父类时,会在父类的虚函数表中查找对应的虚函数。 指向子类时,会在切割后的父类(子类中完成对父类虚函数重写)的虚函数表中查找已经被覆盖的对应的子类的虚函数


总结一下就是,多态调用就是在运行时去虚函数表中找虚函数的地址来进行调用,所以可以达到指向父类调父类,指向子类调子类虚函数。

如果去掉 virtual ,则是普通调用,在编译时通过调用者的类型确定函数的地址

4.3 静态绑定和动态绑定

简单来说,静态就是编译时,动态就是运行时,

静态绑定是在编译时确定程序的行为,也叫静态多态(函数重载),
动态绑定是在程序运行期间确定程序行为

五,单继承和多继承的虚函数表

在单继承和多继承关系中,我们关注的是派生类对象的虚表模型,因为基类的虚表模型前面我们已经看过了,没什么需要特别研究的。

5.1 单继承的虚函数表

看下面的代码:

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;
};

在这里插入图片描述
单继承就是将基类的虚函数表拷贝下来,将自己重写的虚函数覆盖。

5.2 多继承的虚函数表

假设一个派生类继承了两个基类,计算这个派生类的大小

看下面的代码:

class Base1 {
public:virtual void func1() { cout << "Base1::func1" << endl; }virtual void func2() { cout << "Base1::func2" << endl; }
private:int b1;
};
class Base2 {
public:virtual void func1() { cout << "Base2::func1" << endl; }virtual void func2() { cout << "Base2::func2" << endl; }
private:int b2;
};
class Derive : public Base1, public Base2 {
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }
private:int d1;
};int main() {Derive d;return 0;
}

在这里插入图片描述

派生类继承了两个基类的虚表,所以说有两张虚表,并且同时覆盖了重写的虚函数地址,如果派生类有自己的虚函数,那么这个虚函数的地址放在继承的第一张虚表中。

六,继承和多态的常见问题

  1. 内联函数也可以是虚函数,当内联函数是普通调用时,其内联属性还在,当多态调用时,会失去其内联属性。
  2. 静态成员函数不能是虚函数,因为没有this指针,无法访问虚函数表。
  3. 构造函数不能是虚函数,因为虚表指针是在构造函数初始化列表之前初始化的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/739959.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QGIS 开发之旅一《二次开发环境搭建》

1、 安装QT 下载QT Index of /new_archive/qt 我选择的版本是 Qt5.14.2 2、安装VS2017 Downloads & Keys - Visual Studio Subscriptions。下载后选择windows通用平台开发和C 开发就可以了。 3、安装插件QT vs tools 搜索 qt vs tools&#xff0c;选择第一个安装 …

Python合并两张图片 | 先叠透明度再合并 (附Demo)

目录 前言正文 前言 用在深度学习可增加噪音&#xff0c;增加数据集等 推荐阅读&#xff1a;Pytorch 图像增强 实现翻转裁剪色调等 附代码&#xff08;全&#xff09; 正文 使用Pillow库来处理图像&#xff08;以下两张图来自网络&#xff09; 图一&#xff1a; 图二&…

​FastIce-Tech 企业官网开源模版:专为中小企业设计的轻量级网址

标题&#xff1a;FastIce-Tech 企业官网开源模版&#xff1a;专为中小企业设计的轻量级网址 中小企业在建立企业官网时常常面临着时间、资源和技术的限制。为了解决这些问题&#xff0c;FastIce-Tech 企业官网开源模版应运而生。它是一个基于 Vue.js、ElementUI 和 Vue-Router …

(一)运行起自己的chatGPT

一、运行步骤 前面所有步骤可以参见https://datawhaler.feishu.cn/docx/BwjzdQPJRonFh8xeiSOcRUI3n8b 二、注意 需要注意的是&#xff1a; 部署起来后&#xff0c;必须使用域名访问才能进入。用ip地址端口访问不成功 三、运行效果 gradio需要额外配置一个外部端口&#x…

springboot+nacos使用

依赖 nacos服务发现和注册的依赖 <dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId> </dependency><dependency><groupId>com.alibaba.cloud</g…

亿发解析:互联网浪潮席卷,新零售崛起成为未来十年无可忽视之势

随着人们消费能力和水平的提高&#xff0c;消费者对产品质量的关注已不再仅限于产品本身&#xff0c;而更加强调产品质量与消费服务体验的双重重要性。随着互联网、移动支付、快递物流等技术的发展&#xff0c;这些技术催生了零售领域的新模式、新经济和新业态&#xff0c;为新…

机试:最大子序列的和

问题描述: 算法思想: 若第(i-1)个序列的小于0,则第i个序列的最大值为nums[i]; 若第(i-1)个序列的小于0,则第i个序列的最大值为max(i-1) nums[i]; 如果max(i-1)>0,max(i)max(i-1)Nums(i) 如果max(i-1)<0,max(i)Nums(i)代码示例: #include <bits/stdc.h> //该算法…

第五十六回 徐宁教使钩镰枪 宋江大破连环马-飞桨图像分类套件PaddleClas初探

宋江等人学会了钩镰枪&#xff0c;大胜呼延灼。呼延灼损失了很多人马&#xff0c;不敢回京&#xff0c;一个人去青州找慕容知府。一天在路上住店&#xff0c;马被桃花山的人偷走了&#xff0c;于是到了青州&#xff0c;带领官兵去打莲花山。 莲花山的周通打不过呼延灼&#xf…

Promise其实也不难

难点图解&#xff1a;then&#xff08;&#xff09;方法 ES6学习网站&#xff1a;ES6 入门教程 解决&#xff1a;回调地狱&#xff08;回调函数中嵌套回调&#xff09; 两个特点&#xff1a; &#xff08;1&#xff09;对象的状态不受外界影响。Promise对象代表一个异步操作&…

AvP:水平基因转移HGT检测

帮其他人做的一个尝试&#xff0c;本身不太了解这一块&#xff0c;要是做错了请多多包涵 Home GDKO/AvP Wiki GitHub 安装AvP 数据库准备 git clone https://github.com/GDKO/AvP.git conda create --name avp conda activate avp conda install -y -c bioconda mafft bl…

鸿蒙Harmony应用开发—ArkTS声明式开发(基础手势:PluginComponent)

提供外部应用组件嵌入式显示功能&#xff0c;即外部应用提供的UI可在本应用内显示。 说明&#xff1a; 该组件从API Version 9开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。本组件为系统接口。 子组件 无 接口 PluginComponent(value:…

strstr函数及其模拟实现

模拟实现的代码&#xff1a; char* my_strstr(char* p1, char* p2) {char* startp1;//记录被查找字符串的首地址char* begin p2;//记录要查找字符串的首地址while (1){while (*p1 ! *p2)//首元素不相同&#xff0c;p1向后移动一位{p1;}start p1;//找到了首元素相同的地址&am…

​如何防止网络攻击?

应对不同类型网络攻击的最佳途径是“知己”、“知彼”&#xff0c;在了解它们的工作原理、能够识别其手段、方法及意图的前提下&#xff0c;找出针对性的应对文案。今天&#xff0c;就为大家总结以下防止不同类型网络攻击的有效方法&#xff0c;希望无论是对个人、还是企业和组…

在文件夹下快速创建vue项目搭建vue框架详细步骤

一、首先在你的电脑目录下新建一个文件夹 进入该文件夹并打开控制台&#xff08;输入cmd指令&#xff09; 进入控制台后输入 vue create springboot_vue (自己指定名称) 如果出现这类报错如&#xff1a;npm install 的报错npm ERR! network request to http://registry.cnp…

python读取大型csv文件,降低内存占用,提高程序处理速度

文章目录 简介读取前多少行读取属性列逐块读取整个文件总结参考资料 简介 遇到大型的csv文件时&#xff0c;pandas会把该文件全部加载进内存&#xff0c;从而导致程序运行速度变慢。 本文提供了批量读取csv文件、读取属性列的方法&#xff0c;减轻内存占用情况。 import pand…

纯前端Web网页内嵌AutoCAD,支持在线编辑DWG、dxf等文档。

随着企业信息化的发展&#xff0c;越来越多的企业有网页在线浏览和编辑DWG文档&#xff08;AutoCad生成的文档&#xff09;的需求&#xff0c;但是新版浏览器纷纷取消了对NPAPI插件的支持&#xff0c;导致之前一些可以在线在线浏览和编辑DWG文档纷纷失效&#xff0c;今天推荐一…

【掌握版本控制:Git 入门与实践指南】操作仓库文件|分支管理

&#x1f3ac;慕斯主页&#xff1a;修仙—别有洞天 ♈️今日夜电波&#xff1a;泥中に咲く—ウォルピスカーター 0:34━━━━━━️&#x1f49f;──────── 4:46 &#x1f504; ◀️ ⏸ ▶…

揭秘反向代理:探索其神秘之处

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

141 Linux 系统编程18,线程,ps –Lf 进程 查看LWP,线程间共享数据,优缺点,编译加-lpthread,

一 线程概念 什么是线程 LWP&#xff1a;light weight process 轻量级的进程&#xff0c;本质仍是进程(在Linux环境下) 进程&#xff1a;独立地址空间&#xff0c;拥有PCB 线程&#xff1a;有独立的PCB&#xff0c;但没有独立的地址空间(共享) 区别&#xff1a;在于是否共…

【1688运营】如何拆解竞争对手店铺和单品数据?

关注竞争对手数据是1688运营中不可或缺的一环&#xff0c;它有助于企业更好地了解市场环境、发现市场机会、学习成功经验、预测市场变化以及提升竞争力。以下是一些建议&#xff0c;帮助你全面、深入地分析竞争对手的店铺和单品数据&#xff1a; 1、监控店铺数据 可以通过店雷…