C++面向对象之多态性
文章目录
- C++面向对象之多态性
- 1.静态多态
- 2.动态多态
- 3.多态的好处
- 3.1使用方法
- 4.纯虚函数
- 5.虚析构与纯虚析构
- 5.1问题
- 5.2解决
- 6.其他知识点
- 7.代码
- 8.测试结果
- 8.1父类中无虚函数,父类的指针指向子类对象,将调用父类中的函数,无法调用子类中的重写函数,即无法实现多态
- 8.2父类中的函数由虚函数实现
- 8.3父类指针对象,无法释放子类中开辟的堆空间数据
- 8.4将父类的析构函数定义为虚析构函数
- 8.5将父类中的析构函数定义为纯虚析构函数
C++面向对象之多态性
1.静态多态
通过函数重载和运算符重载实现,在编译时确定函数早绑定
2.动态多态
派生类和虚函数实现运行时多态,运行阶段实现函数晚绑定。
当子类重写了父类的虚函数时,子类中的虚函数表将替换父类中的虚函数,
所以当父类的引用或者指针指向子类对象时,将发生多态
3.多态的好处
结构清晰
可读性很强
利于后期的拓展和维护
3.1使用方法
父类的引用或者指针指向子类的对象
4.纯虚函数
定义格式:virtual 返回值类型 函数名称(参数列表) = 0;
类中有了纯虚函数,称为抽象类,抽象类无法实例化对象
子类必须重写父类中的所有纯虚函数,才能实例化对象,否则也是抽象类
5.虚析构与纯虚析构
5.1问题
多态使用时,如果子类中有属性开辟到了堆区,则父类指针释放时无法调用子类的析构函数
5.2解决
将父类中的析构函数改为虚析构函数或者纯虚析构函数
纯虚析构函数需要先声明,后实现,但是纯虚函数可以只声明
如果子类没有堆区数据,可以不将父类的析构函数写成虚析构或者纯虚析构
拥有纯虚析构函数的类也属于抽象类
6.其他知识点
空类(即里面什么代码都没有)的大小通过sizeof测定的结果是1
vfptr(虚函数表指针) ——> vftable(虚函数表,用于记录虚函数的地址)
7.代码
#include<iostream>using namespace std;class Animal {
public:Animal() {cout << "基类中的构造函数被调用!" << endl;}// 虚函数,虚函数将实现晚绑定virtual void speack() {cout << "动物在说话!" << endl;}virtual ~Animal() = 0; // 纯虚析构函数,纯虚析构声明后,必须定义
};// Animal中的纯虚析构实现部分,可以在这里释放父类开辟的堆空间
Animal::~Animal() {}class Cat :public Animal {public:Cat() {cout << "cat中的构造函数被调用" << endl;}void speack() {cout << "小猫在说话!" << endl;}~Cat() {cout << "cat中的析构函数被调用" << endl;}
};class Dog : public Animal {
public:void speack() {cout << "小狗在说话" << endl;}~Dog() {cout << "dog中的析构函数被调用" << endl;}
};// 地址早绑定,在编译阶段函数地址已经早绑定Animal
void doSpeack(Animal& animal) {animal.speack();
}// 派生类重写基类中的虚函数,当基类的引用或者指针指向派生类时,实现多态
void test01() {Cat cat;Dog dog;doSpeack(cat);doSpeack(dog);}// 基类和派生类的构造函数和析构函数的调用
//
void test02() {Animal* p_cat = new Cat(); // 开辟内存到堆区,需要手动释放if (p_cat != NULL) {delete p_cat; // 释放指针p_cat = NULL;}}// --------------------------------------------------------------------------
class CPU {
public:virtual void caculate() = 0;
};class VideoCard {
public:virtual void display() = 0;
};class Memory {
public:virtual void storage() = 0;
};class Computer {
public:Computer(CPU *cpu, VideoCard *videoCard,Memory * memory) {this -> m_cpu = cpu;this -> m_vc = videoCard;this -> m_mem = memory;}// 电脑的工作void work() {m_cpu->caculate();m_vc->display();m_mem->storage();}~Computer() {if (m_cpu != NULL) {delete m_cpu;}if (m_vc != NULL) {delete m_vc;}if (m_mem != NULL) {delete m_mem;}}private:CPU* m_cpu;VideoCard* m_vc;Memory* m_mem;
};
// inter厂商
class InterCPU :public CPU {public:void caculate() {cout << "inter的CPU开始计算了" << endl;}
};class InterVideoCard :public VideoCard {public:void display() {cout << "inter的显卡开始计算了" << endl;}
};
class InterMemory :public Memory {public:void storage() {cout << "inter的内存条开始存储了" << endl;}
};
// lenovoclass lenovoCPU :public CPU {public:virtual void caculate() {cout << "lenovo的CPU开始计算了" << endl;}
};class lenovoVideoCard :public VideoCard {public:virtual void display() {cout << "lenovo的显卡开始计算了" << endl;}
};
class lenovoMemory :public Memory {public:virtual void storage() {cout << "lenovo的内存条开始存储了" << endl;}
};void test03() {// 创建电脑CPU* interCPU = new InterCPU;VideoCard* interCard = new InterVideoCard;Memory* interMemory = new InterMemory;Computer* computer1 = new Computer(interCPU, interCard, interMemory);computer1->work();delete computer1;cout << "---------------------------" << endl;Computer* computer2 = new Computer(new lenovoCPU, new lenovoVideoCard,new lenovoMemory);computer2->work();delete computer2;cout << "---------------------------" << endl;Computer* computer3 = new Computer(new InterCPU, new lenovoVideoCard, new lenovoMemory);computer3->work();delete computer3;
}
int main() { //test01();//test02();test03();system("pause");return 0;
}
8.测试结果
8.1父类中无虚函数,父类的指针指向子类对象,将调用父类中的函数,无法调用子类中的重写函数,即无法实现多态

8.2父类中的函数由虚函数实现

父类的指针指向小猫子类,将调用小猫类中的重写函数,实现多态
8.3父类指针对象,无法释放子类中开辟的堆空间数据

cat类中的析构函数没有被调用,无法释放开辟的堆空间数据
8.4将父类的析构函数定义为虚析构函数
子类中的析构函数被调用

8.5将父类中的析构函数定义为纯虚析构函数
子类中的析构函数被调用,将父类中的析构函数声明为纯虚析构函数后,必须要实现声明的函数。

相关文章:
C++面向对象之多态性
文章目录C面向对象之多态性1.静态多态2.动态多态3.多态的好处3.1使用方法4.纯虚函数5.虚析构与纯虚析构5.1问题5.2解决6.其他知识点7.代码8.测试结果8.1父类中无虚函数,父类的指针指向子类对象,将调用父类中的函数,无法调用子类中的重写函数&…...
Android性能优化系列篇:弱网优化
弱网优化1、Serializable原理通常我们使用Java的序列化与反序列化时,只需要将类实现Serializable接口即可,剩下的事情就交给了jdk。今天我们就来探究一下,Java序列化是怎么实现的,然后探讨一下几个常见的集合类,他们是…...
Mysql 插入大批量数据调优方法
Mysql 插入大批量数据调优方法[toc]1、多线程插入(单表)在数据里做插入操作的时候,整体时间的分配是这样的:链接耗时 (30%)发送query到服务器 (20%)解析query (20%&#…...
matlab基础
系列文章目录 文章目录系列文章目录前言1 基本用法总结基础语法桌面管理矩阵均匀间隔矢量矩阵创建矩阵索引前言 介绍了matlab的基本用法 1 基本用法 >> save filename.mat % 将当前工作区的所有变量保存为mat文件 >> load filename.mat % 加载文件>> loa…...
自动化测试——多窗口切换和切换frame
这里写目录标题一、多窗口切换1、base.py:公共代码2、切换句柄的方式1,通过for循环3、切换句柄的方式2,通过索引切换4、源代码二、frame窗口1、什么是frame?2、Frame 分类3、判断要定位的元素在不在frame中两种方式方式一:鼠标选…...
C#中,Elasticsearch.Net判断空字符串
之前有个业务需求,由于最开始存储到es里的,是默认空字符串。 后面程序取数据时,发现需要取空字符串的数据时,不好取出来。 字符串的字段如图: 实际数据如图: 我用的是C#语言,使用的是Elastic…...
23种设计模式-适配器模式
适配器模式(Adapter Pattern)是一种常用的设计模式,它可以将不兼容的接口转换成可兼容的接口,使得原本不能一起工作的类可以协同工作。 在Java中,适配器模式一般有两种实现方式,即类适配器模式和对象适配器…...
深入理解this指向问题
this指向 在运行时绑定,所以this和函数的调用方式和调用的位置有关,和定义的位置没关系 绑定规则 默认绑定(非严格模式下this指向全局变量,在严格模式下函数内的this指向undefined) 独立函数调用,没有主题 …...
事业单位联考(综合应用A类)典型例题教案
【联考A类】根据材料2,请你概括C市B县旅游质监所投诉处理科小王在接待投诉时存在的主要问题,并指出问题的具体表现。(35分)要求:准确、全面、分条作答。字数在300字以内。材料2:某日,几位游客家…...
frp内网穿透实验
Frp (Fast Reverse Proxy) 是比较流行的一款。FRP 是一个免费开源的用于内网穿透的反向代理应用,它支持 TCP、UDP 协议, 也为 http 和 https 协议提供了额外的支持。你可以粗略理解它是一个中转站, 帮你实现 公网 ←→ FRP(服务器) ←→ 内网…...
认识JavaScript中的防抖函数
👨 作者简介:大家好,我是Taro,前端领域创作者 ✒️ 个人主页:唐璜Taro 🚀 支持我:点赞👍📝 评论 ⭐️收藏 文章目录前言一、防抖是什么?1. deounce-v1的基本…...
macOS 13.3 Beta 2 (22E5230e)With OpenCore 0.8.9正式版 and winPE双引导分区原版镜像
原文地址:http://www.imacosx.cn/112340.html,转载需注明出处镜像特点完全由黑果魏叔官方制作,针对各种机型进行默认配置,让黑苹果安装不再困难。系统镜像设置为双引导分区,全面去除clover引导分区(如有需要…...
JetPack—DataStore核心原理与使用
简介 首先,DataStore是Jetpack一部分,是一种数据存储解决方案。其次,DataStore使用协程及flow以异步、一致的方式实现数据的存储。最后是DataStore的实现,分为Preferences DataStore和Proto DataStore:Preferences Da…...
热烈祝贺|酒事有鲤盛装亮相2023中国(山东)精酿啤酒产业发展创新论坛暨展览会
酒事有鲤(济南)品牌管理有限公司是一家致力于将世界顶级精酿啤酒技术和理念与“ 在地”文化有机融合,做世界认 可的多元化好啤酒,通过精致 舒适的家门口酒馆,让啤酒的 世界观更为完整。 中国生物发酵产业协会联合齐鲁…...
深度强化学习DLR
1 强化学习基础知识 强化学习过程:⾸先环境(Env)会给智能体(Agent)⼀个状态(State),智能体接收到环境给的观测值之后会做出⼀个动作(Action),环境接收到智能体给的动作之后会做出⼀系列的反应,例如对这个动作给予⼀个奖励(Reward…...
Android Handler机制(四) Message源码分析
一. 简介 接上一篇文章:Android Handler机制(三) Looper源码分析 ,我们来继续分析一下Message源码 这一系列文章都是为了深入理解Handler机制. Message 作为消息传递的载体,源码主要分为以下 几个部分: 1. 操作数据相关,类似 getter()和 setter()这种…...
【Git】git命令(全)
Git1、本地操作2、版本管理3、远端仓库4、分支管理5、缓存stash6、遗留rebase7、标签管理8、解决冲突9、参考教程10、示例代码1、本地操作 Linux安装git:yum install git查看git版本 git version查看git设置 git config --list设置git属性 git config --global初始…...
软考论文-成本管理(1)
成本管理 1.成本管理的主要内容? 规划成本:制定一个成本管理的计划。估算成本:根据项目范围说明书,项目管理计划和wbs等文档,采用xxx方法进行估算成本成本预算:可以算工作包的费用,制定预算和…...
Java 多线程 --- 锁的概念和类型划分
Java 多线程 --- 锁的概念和类型划分锁的概念乐观锁与悲观锁公平锁与非公平锁什么是可重入锁独占锁与共享锁轻量级锁和重量级锁自旋锁 (Spinlock)锁的概念 锁可以将多个线程对共享数据的并发访问转换为串行访问, 这样一个共享数据一次只能被一个线程访问, 该线程访问结束后其他…...
python程序员狂飙上头——京海市大嫂单推人做个日历不过分吧?
嗨害大家好鸭!我是小熊猫~ 这个反黑剧其实火了很久了, 但是我现在才有空开始看 该说不说,真的很上头!!! 大嫂简直就像是干枯沙漠里的玫瑰 让人眼前一亮哇~~ 我小熊猫此时此刻就成为大嫂的单推人&…...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
sqlserver 根据指定字符 解析拼接字符串
DECLARE LotNo NVARCHAR(50)A,B,C DECLARE xml XML ( SELECT <x> REPLACE(LotNo, ,, </x><x>) </x> ) DECLARE ErrorCode NVARCHAR(50) -- 提取 XML 中的值 SELECT value x.value(., VARCHAR(MAX))…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...
【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)
升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点,但无自动故障转移能力,Master宕机后需人工切换,期间消息可能无法读取。Slave仅存储数据,无法主动升级为Master响应请求ÿ…...
嵌入式学习笔记DAY33(网络编程——TCP)
一、网络架构 C/S (client/server 客户端/服务器):由客户端和服务器端两个部分组成。客户端通常是用户使用的应用程序,负责提供用户界面和交互逻辑 ,接收用户输入,向服务器发送请求,并展示服务…...
使用Spring AI和MCP协议构建图片搜索服务
目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式(本地调用) SSE模式(远程调用) 4. 注册工具提…...
Sklearn 机器学习 缺失值处理 获取填充失值的统计值
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 使用 Scikit-learn 处理缺失值并提取填充统计信息的完整指南 在机器学习项目中,数据清…...
ThreadLocal 源码
ThreadLocal 源码 此类提供线程局部变量。这些变量不同于它们的普通对应物,因为每个访问一个线程局部变量的线程(通过其 get 或 set 方法)都有自己独立初始化的变量副本。ThreadLocal 实例通常是类中的私有静态字段,这些类希望将…...
