当前位置: 首页 > 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、信号是否携带…...

React 第五十五节 Router 中 useAsyncError的使用详解

前言 useAsyncError 是 React Router v6.4 引入的一个钩子&#xff0c;用于处理异步操作&#xff08;如数据加载&#xff09;中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误&#xff1a;捕获在 loader 或 action 中发生的异步错误替…...

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

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

Zustand 状态管理库:极简而强大的解决方案

Zustand 是一个轻量级、快速和可扩展的状态管理库&#xff0c;特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...

《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》

引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...

Psychopy音频的使用

Psychopy音频的使用 本文主要解决以下问题&#xff1a; 指定音频引擎与设备&#xff1b;播放音频文件 本文所使用的环境&#xff1a; Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...

AI病理诊断七剑下天山,医疗未来触手可及

一、病理诊断困局&#xff1a;刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断"&#xff0c;医生需通过显微镜观察组织切片&#xff0c;在细胞迷宫中捕捉癌变信号。某省病理质控报告显示&#xff0c;基层医院误诊率达12%-15%&#xff0c;专家会诊…...

视觉slam十四讲实践部分记录——ch2、ch3

ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...

CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)

漏洞概览 漏洞名称&#xff1a;Apache Flink REST API 任意文件读取漏洞CVE编号&#xff1a;CVE-2020-17519CVSS评分&#xff1a;7.5影响版本&#xff1a;Apache Flink 1.11.0、1.11.1、1.11.2修复版本&#xff1a;≥ 1.11.3 或 ≥ 1.12.0漏洞类型&#xff1a;路径遍历&#x…...

日常一水C

多态 言简意赅&#xff1a;就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过&#xff0c;当子类和父类的函数名相同时&#xff0c;会隐藏父类的同名函数转而调用子类的同名函数&#xff0c;如果要调用父类的同名函数&#xff0c;那么就需要对父类进行引用&#…...

零知开源——STM32F103RBT6驱动 ICM20948 九轴传感器及 vofa + 上位机可视化教程

STM32F1 本教程使用零知标准板&#xff08;STM32F103RBT6&#xff09;通过I2C驱动ICM20948九轴传感器&#xff0c;实现姿态解算&#xff0c;并通过串口将数据实时发送至VOFA上位机进行3D可视化。代码基于开源库修改优化&#xff0c;适合嵌入式及物联网开发者。在基础驱动上新增…...