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

【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. 构造函数不能是虚函数,因为虚表指针是在构造函数初始化列表之前初始化的

相关文章:

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

C多态概念详解 一&#xff0c;多态概念二&#xff0c;多态的定义2.1 多态构成的条件2.2 什么是虚函数2.3 虚函数的重写2.3.1 虚函数重写的特例2.3.2 override和final 2.4 重载和重写&#xff08;覆盖&#xff09;和重定义&#xff08;隐藏&#xff09;的区别 三&#xff0c;抽象…...

Python 导入Excel三维坐标数据 生成三维曲面地形图(面) 2、线条平滑曲面但有间隔

环境和包: 环境 python:python-3.12.0-amd64包: matplotlib 3.8.2 pandas 2.1.4 openpyxl 3.1.2 scipy 1.12.0 代码: import pandas as pd import matplotlib.pyplot as plt from mpl_toolkits.mplot3d import Axes3D from scipy.interpolate import griddata imp…...

前端精准测试调用链路分析

精准测试在评估需求的测试范围时&#xff0c;需要评估一下代码的影响范围&#xff0c;这个范围有两部分&#xff1a;一是需求直接修改的代码&#xff1b;二是修改代码影响到的功能模块。代码影响到的功能一般是通过调用链路分析来实现的&#xff0c;java和kotlin代码可以由java…...

Objective-C blocks 概要

1.block的使用 1.1什么是block&#xff1f; Blocks是C语言的扩充功能&#xff1a;带有自动变量&#xff08;局部变量&#xff09;的匿名函数。 “带有自动变量”在Blocks中表现为“截取自动变量" “匿名函数”就是“不带名称的函数” 块&#xff0c;封装了函数调用及调用…...

Linux操作系统-07-Linux安装应用

一、使用rpm安装应用&#xff08;不推荐&#xff09; 先下载到本地&#xff0c;以.rpm文件名结尾&#xff0c;下载完成后&#xff0c;再安装 rpm -qa | grep mysql #查询当前系统是否有下载过mysql包 先上传mysql的rpm安装包到linux的opt目录 安装 rpm -ivh …...

DevOps实战:Docker、Kubernetes与Jenkins的完美融合

DevOps与容器化技术&#xff1a;Docker、Kubernetes和Jenkins 引言 在软件开发领域&#xff0c;DevOps文化和容器化技术已经成为当今最热门的话题之一。DevOps的目标是缩短开发和运维之间的距离&#xff0c;提高软件交付的速度和质量。而容器化技术&#xff0c;如Docker和Kub…...

Python面向对象——程序架构

需求 创建图形管理器 -记录多种图形(圆形、矩形.) --提供计算总面积的方法&#xff0c; 要求:增加新图形&#xff0c;不影响图形管理器 测试: 创建图形管理器&#xff0c;存储多个图形对象。 通过图形管理器&#xff0c;调用计算总面积方法 思路 ​​​​​​​ 代码 # ------…...

springboot单体项目链路日志跟踪及接口耗时

最近接触一个新的传统项目,在联调过程中,查看日志特别不方便,既无trackId,即无接口耗时,所以写了该博客。话不多说,直接上代码 1、实体类user package com.yk.domain;import lombok.Data;@Data public class User {private Long id;private String username;private St…...

力扣hot---岛屿数量

dfs思路&#xff1a; 首先通过两层for循环遍历每一个点&#xff0c;如果这个点为0或者2&#xff08;这个2是什么呢&#xff1f;是在遍历该点以及该点连成的这一片区域中&#xff0c;因为通过深度优先搜索&#xff0c;遍历该点就等于遍历这一片区域&#xff0c;遍历这篇区域中的…...

如何在Linux使用docker安装Plik并实现无公网ip上传下载内网存储的文件资源

文章目录 1. Docker部署Plik2. 本地访问Plik3. Linux安装Cpolar4. 配置Plik公网地址5. 远程访问Plik6. 固定Plik公网地址7. 固定地址访问Plik 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的 人工智能学习网站&#xff0c; 通俗易懂&#xff0c;风趣幽默&…...

Nginx反向代理详解

1. 什么是反向代理 反向代理是一种服务器代理的方式&#xff0c;它代理了客户端的请求并将请求转发给后端服务器&#xff0c;然后将后端服务器的响应返回给客户端。在这个过程中&#xff0c;客户端并不直接与后端服务器通信&#xff0c;而是通过反向代理服务器来实现请求转发和…...

【Android】 ClassLoader 知识点提炼

1.Java中的 ClassLoader 1.1 、ClassLoader的类型 Java 中的类加载器主要有两种类型&#xff0c;即系统类加载器和自定义类加载器。其中系统类 加载器包括3种&#xff0c;分别是 Bootstrap ClassLoader、Extensions ClassLoader 和 Application ClassLoader。 1.1.1.Bootstra…...

16. C++标准库

C标准库兼容C语言标准函数库&#xff0c;可以在C标准库中直接使用C语言标准函数库文件&#xff0c;同时C标准库增加了自己的源代码文件&#xff0c;新增文件使用C编写&#xff0c;多数代码放在std命名空间中&#xff0c;所以连接C标准库文件后还需要 using namespace std;。 【…...

JVM内存结构介绍

1. 什么是JVM 我们都知道在 Windows 系统上一个软件包装包是 exe 后缀的&#xff0c;而这个软件包在苹果的 Mac OSX 系统上是无法安装的。类似地&#xff0c;Mac OSX 系统上软件安装包则是 dmg 后缀&#xff0c;同样无法在 Windows 系统上安装。 Java 代码为什么可以在 Windows…...

Linux常见指令总结

ls&#xff1a;显示当前目录下文件列表 常用的命令行参数&#xff1a; -l 显示更多的文件属性 -a 显示所有的文件/目录&#xff08;包括隐藏的&#xff09; -d 只显示目录 ps&#xff1a;参数可以叠加使用。 例如&#xff1a;ls -la 显示所有文件…...

Day35-Linux网络管理5

Day35-Linux网络管理5 1. 网卡配置2. DNS客户端域名解析配置3. 给网卡配多个IP4. ip地址查看和设置4.1 ifconfig命令4.2 ip命令4.3 ip命令&#xff1a;查看和设置网络配置4.4 ip命令帮助 5. 路由5.1 路由功能分类&#xff1a;5.2 查看路由&#xff1a;5.3 路由表&#xff1a;5.…...

9个神奇免费AI编程助手,实现高效自动代码生成!

在AIGC技术工具快速发展的时代&#xff0c;对高效智能编程工具的需求和关注已达到空前的高度。本文将介绍9款免费且好用的AI编程助手工具。无论你是经验丰富的开发人员还是刚开始编程旅程的新手&#xff0c;这些AI代码软件都能帮助你提高项目开发的生产力、创造力和准确性&…...

Python 导入Excel三维坐标数据 生成三维曲面地形图(体) 5-3、线条平滑曲面且可通过面观察柱体变化(三)

环境和包: 环境 python:python-3.12.0-amd64包: matplotlib 3.8.2 pandas 2.1.4 openpyxl 3.1.2 scipy 1.12.0 代码: import pandas as pd import matplotlib.pyplot as plt from mpl_toolkits.mplot3d import Axes3D from scipy.interpolate import griddata fro…...

【CSP】2022–09-3 防疫大数据 100分 STL大模拟 使用map优化索引 有坑得注意

2022–09-3 防疫大数据 STL大模拟 使用map优化索引 2022–09-3 防疫大数据 STL大模拟 使用map优化索引基本思路遇到的问题&#xff08;学到的东西&#xff09;感悟完整代码 2022–09-3 防疫大数据 STL大模拟 使用map优化索引 这题中规中矩&#xff0c;不算太难也不算太简单&am…...

【Linux基础(三)】信号

学习分享 1、信号的基本概念2、查看信号列表3、常见信号名称4、signal库函数5、发送信号kill6、kill - signal &#xff08;无参信号&#xff09;示例6.1、kill - signal (不可靠信号)示例6.2、kill - signal (可靠信号)示例 7、信号分类7.1、信号运行原理分类7.2、信号是否携带…...

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

RocketMQ延迟消息机制

两种延迟消息 RocketMQ中提供了两种延迟消息机制 指定固定的延迟级别 通过在Message中设定一个MessageDelayLevel参数&#xff0c;对应18个预设的延迟级别指定时间点的延迟级别 通过在Message中设定一个DeliverTimeMS指定一个Long类型表示的具体时间点。到了时间点后&#xf…...

java_网络服务相关_gateway_nacos_feign区别联系

1. spring-cloud-starter-gateway 作用&#xff1a;作为微服务架构的网关&#xff0c;统一入口&#xff0c;处理所有外部请求。 核心能力&#xff1a; 路由转发&#xff08;基于路径、服务名等&#xff09;过滤器&#xff08;鉴权、限流、日志、Header 处理&#xff09;支持负…...

微软PowerBI考试 PL300-选择 Power BI 模型框架【附练习数据】

微软PowerBI考试 PL300-选择 Power BI 模型框架 20 多年来&#xff0c;Microsoft 持续对企业商业智能 (BI) 进行大量投资。 Azure Analysis Services (AAS) 和 SQL Server Analysis Services (SSAS) 基于无数企业使用的成熟的 BI 数据建模技术。 同样的技术也是 Power BI 数据…...

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列&#xff0c;以便知晓哪些列包含有价值的数据&#xff0c;…...

服务器--宝塔命令

一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行&#xff01; sudo su - 1. CentOS 系统&#xff1a; yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...

GO协程(Goroutine)问题总结

在使用Go语言来编写代码时&#xff0c;遇到的一些问题总结一下 [参考文档]&#xff1a;https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现&#xff1a; 今天在看到这个教程的时候&#xff0c;在自己的电…...

WPF八大法则:告别模态窗口卡顿

⚙️ 核心问题&#xff1a;阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程&#xff0c;导致后续逻辑无法执行&#xff1a; var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题&#xff1a…...

抽象类和接口(全)

一、抽象类 1.概念&#xff1a;如果⼀个类中没有包含⾜够的信息来描绘⼀个具体的对象&#xff0c;这样的类就是抽象类。 像是没有实际⼯作的⽅法,我们可以把它设计成⼀个抽象⽅法&#xff0c;包含抽象⽅法的类我们称为抽象类。 2.语法 在Java中&#xff0c;⼀个类如果被 abs…...