【C++】类和对象-多态

1.多态的基本语法

在这里插入图片描述
在这里插入图片描述
代码

#include <iostream>
using namespace std;
/******************************************/
class Animal {
public://speak函数就是虚函数//函数前面加上virtual关键字,变成虚函数,//那么编译器在编译的时候就不能确定函数调用了。virtual void speak() {cout << "动物在说话" << endl;}
};class Cat:public Animal {
public:void speak() {cout << "小猫在说话" << endl;}
};class Dog :public Animal {
public:void speak() {cout << "小狗在说话" << endl;}
};//我们希望传入什么对象,那么就调用什么对象的函数
//如果函数地址在编译阶段就能确定,那么静态联编
//如果函数地址在运行阶段才能确定,那么动态联编
void doSpeak(Animal& animal) {animal.speak();
}//多态满足条件:
//1、有继承关系
//2、子类重写父类中的虚函数
//多态使用:
//父类指针或引用指向子类对象void test01() {Cat cat;doSpeak(cat);Dog dog;doSpeak(dog);
}
/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

结果
在这里插入图片描述

2.多态的原理剖析

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

3.多态案例一-计算器类

在这里插入图片描述
在这里插入图片描述

(1)普通实现代码:

#include <iostream>
using namespace std;
/******************************************/
//普通实现
class Calculator {
public:int getResult(string oper) {if (oper == "+") {return m_Num1 + m_Num2;}else if(oper == "-") {return m_Num1 - m_Num2;}else if (oper == "*") {return m_Num1 * m_Num2;}//如果要提供新的运算,需要修改源码}
public:int m_Num1;int m_Num2;
};void test01() {//普通实现测试Calculator cal;cal.m_Num1 = 100;cal.m_Num2 = 100;cout << cal.m_Num1 << " + " << cal.m_Num2 << " = " << cal.getResult("+") << endl;cout << cal.m_Num1 << " - " << cal.m_Num2 << " = " << cal.getResult("-") << endl;cout << cal.m_Num1 << " * " << cal.m_Num2 << " = " << cal.getResult("*") << endl;
}
/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

代码结果
在这里插入图片描述

(2)多态实现代码:

#include <iostream>
using namespace std;
/******************************************/
//多态实现//抽象计算器类
//多态优点:代码组织结构清晰,可读性强,利于前期和后期的扩展以及维护
class AbstractCalculator {
public:virtual int getResult() {return 0;}int m_Num1;int m_Num2;
};//加法计算器类
class AddCalculator :public AbstractCalculator {
public:int getResult() {return m_Num1 + m_Num2;}
};//减法计算器类
class SubCalculator :public AbstractCalculator {
public:int getResult() {return m_Num1 - m_Num2;}
};//乘法计算器类
class MulCalculator :public AbstractCalculator {
public:int getResult() {return m_Num1 * m_Num2;}
};void test01() {//创建加法计算器AbstractCalculator* abc = new AddCalculator;abc->m_Num1 = 100;abc->m_Num2 = 100;cout << abc->m_Num1 << " + " << abc->m_Num2 << " = " << abc->getResult() << endl;delete abc;//用完记得销毁//创建减法计算器abc = new SubCalculator;abc->m_Num1 = 10;abc->m_Num2 = 10;cout << abc->m_Num1 << " - " << abc->m_Num2 << " = " << abc->getResult() << endl;delete abc;//用完记得销毁//创建减法计算器abc = new MulCalculator;abc->m_Num1 = 10;abc->m_Num2 = 10;cout << abc->m_Num1 << " * " << abc->m_Num2 << " = " << abc->getResult() << endl;delete abc;//用完记得销毁}
/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

结果
在这里插入图片描述
以下是没有加virtual的情况,不是虚函数,不能形成多态,会运行AbstractCalculator类里面的getResult()函数。
在这里插入图片描述

4.纯虚函数和抽象类

在这里插入图片描述

#include <iostream>
using namespace std;
/******************************************/
//纯虚函数和抽象类
class Base {
public://纯虚函数//类中只要有一个纯虚函数就称为抽象类//抽象类无法实例化对象//子类必须重写父类中的纯虚函数,否则也属于抽象类virtual void func() = 0;
};class Son :public Base {
public:virtual void func() {cout << "func()调用" << endl;}
};void test01() {//Base base;//错误,抽象类无法实例化对象Base* base = NULL;//base = new Base;//错误,抽象类无法实例化对象base = new Son;base->func();delete base;//记得销毁
}
/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

结果
在这里插入图片描述

5.多态案例二-制作饮品

在这里插入图片描述

#include <iostream>
using namespace std;
/******************************************/
class AbstractDrinking {
public://煮水virtual void boilWater() = 0;//冲泡virtual void brew() = 0;//倒入杯中virtual void putInCup() = 0;//加入辅料virtual void putSomething() = 0;//制作饮品void makingDrink() {boilWater();brew();putInCup();putSomething();}
};//制作咖啡
class MakeCoffee :public AbstractDrinking {//煮水virtual void boilWater() {cout << "煮矿泉水" << endl;}//冲泡virtual void brew() {cout << "冲泡" << endl;}//倒入杯中virtual void putInCup() {cout << "倒入杯子" << endl;}//加入辅料virtual void putSomething() {cout << "加入蜂蜜和牛奶" << endl;}
};//制作茶水
class MakeTea:public AbstractDrinking {//煮水virtual void boilWater() {cout << "煮自来水" << endl;}//冲泡virtual void brew() {cout << "冲泡" << endl;}//倒入杯中virtual void putInCup() {cout << "倒入杯子" << endl;}//加入辅料virtual void putSomething() {cout << "倒入柠檬" << endl;}
};//制作函数
void doWork(AbstractDrinking* drinking) {drinking->makingDrink();delete drinking;
}void test01() {doWork(new MakeCoffee);cout << endl << "________________________" << endl << endl;doWork(new MakeTea);
}
/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

在这里插入图片描述

6.虚析构和纯虚析构

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
代码:

#include <iostream>
using namespace std;
/******************************************/
//虚析构和纯虚析构
class Animal {
public:Animal() {cout << "Animal构造函数" << endl;}virtual void Speak() = 0;//析构函数前面加上virtual关键字,变成虚析构函数//virtual ~Animal() {//	cout << "Animal虚析构函数" << endl;//}virtual ~Animal() = 0;
};Animal::~Animal()
{cout << "Animal纯虚析构函数" << endl;
}
//和包含普通纯虚函数的类一样,包含了纯虚析构函数的类也是一个抽象类,不能够被实例化。class Cat :public Animal {
public:Cat(string name) {cout << "Cat构造函数" << endl;this->m_Name = new string(name);}virtual void Speak() {cout << *this->m_Name << "小猫在说话" << endl;}~Cat() {cout << "Cat析构函数" << endl;if (this->m_Name != NULL) {delete m_Name;this->m_Name = NULL;}}string* m_Name;
};void test01() {Animal* animal = new Cat("Tom");animal->Speak();//通过父类指针去释放,会导致子类对象可能清理不干净,造成内存泄漏//怎么解决?给基类增加一个虚析构函数//虚析构函数就是用来解决: 通过父类指针释放子类对象delete animal;
}/******************************************/
int main()
{/******************************************/test01();/******************************************/system("pause");return 0;
}

结果
在这里插入图片描述

7.多态案例三-电脑组装

在这里插入图片描述

在这里插入图片描述
代码

#include <iostream>
using namespace std;//抽象不同零件类
//抽象CPU类
class CPU {
public://抽象的计算函数virtual void calculate() = 0;
};
//抽象显卡类
class VideoCard {
public://抽象的显示函数virtual void display() = 0;
};
//抽象内存条类
class Memory {
public://抽象的存储函数virtual void storage() = 0;
};
//抽象电脑类
class Computer {
public:Computer(CPU* cpu, VideoCard* vc, Memory* mem) {m_cpu = cpu;m_vc = vc;m_mem = mem;}//提供工作的函数void work() {//让零件工作起来,调用接口m_cpu->calculate();m_vc->display();m_mem->storage();}//提供析构函数 释放3个电脑零件~Computer() {//释放CPU零件if (m_cpu != NULL) {delete m_cpu;m_cpu = NULL;}//释放显卡零件if (m_vc != NULL) {delete m_vc;m_vc = NULL;}//释放内存条零件if (m_mem != NULL) {delete m_mem;m_mem = NULL;}}//!!!!不用虚析构,因为子类(当然啦,压根没有子类)没有成员在堆区开辟数据,不需要执行子类析构函数private:CPU* m_cpu;//CPU的零件指针VideoCard* m_vc;//显卡零件指针Memory* m_mem;//内存条零件指针
};//具体厂商
//Intel厂商
class IntelCPU: public CPU {virtual void calculate() {cout << "IntelCPU开始计算了" << endl;}
};
class IntelVideoCard: public VideoCard {virtual void display() {cout << "Intel显卡开始显示了" << endl;}
};
class IntelMemory : public Memory {virtual void storage() {cout << "Intel内存条开始存储了" << endl;}
};
//Lenovo厂商
class LenovoCPU : public CPU {virtual void calculate() {cout << "LenovoCPU开始计算了" << endl;}
};
class LenovoVideoCard : public VideoCard {virtual void display() {cout << "Lenovo显卡开始显示了" << endl;}
};
class LenovoMemory : public Memory {virtual void storage() {cout << "Lenovo内存条开始存储了" << endl;}
};void test01() {cout << "---------创建第二台电脑------------" << endl;//第一台电脑零件CPU* intelCPU = new IntelCPU;VideoCard* intelCard = new IntelVideoCard;Memory* intelMem = new IntelMemory;//创建第一台电脑Computer* computer1 = new Computer(intelCPU, intelCard, intelMem);computer1->work();delete computer1;cout << endl << "---------创建第二台电脑------------" << endl;//创建第二台电脑Computer* computer2 = new Computer(new IntelCPU, new IntelVideoCard, new IntelMemory);computer2->work();delete computer2;
}int main() {test01();system("pause");return 0;
}

结果
在这里插入图片描述

错误总结:
在这里插入图片描述
这两个名字不能一样,原因如下:
在这里插入图片描述

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

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

相关文章

DevExpress WPF Tree List组件,让数据可视化程度更高!(二)

DevExpress WPF Tree List组件是一个功能齐全、数据感知的TreeView-ListView混合体&#xff0c;可以把数据信息显示为REE、GRID或两者的组合&#xff0c;在数据绑定或非绑定模式下&#xff0c;具有完整的数据编辑支持。 在上文中&#xff08;点击这里回顾DevExpress WPF Tree …

中介者模式——协调多个对象之间的交互

1、简介 1.1、概述 如果在一个系统中对象之间的联系呈现为网状结构&#xff0c;如下图所示&#xff1a; 对象之间存在大量的多对多联系&#xff0c;将导致系统非常复杂&#xff0c;这些对象既会影响别的对象&#xff0c;也会被别的对象所影响&#xff0c;这些对象称为同事对…

LeetCode--HOT100题(21)

目录 题目描述&#xff1a;240. 搜索二维矩阵 II&#xff08;中等&#xff09;题目接口解题思路代码 PS: 题目描述&#xff1a;240. 搜索二维矩阵 II&#xff08;中等&#xff09; 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&am…

【雕爷学编程】MicroPython动手做(28)——物联网之Yeelight 2

知识点&#xff1a;什么是掌控板&#xff1f; 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片&#xff0c;支持WiFi和蓝牙双模通信&#xff0c;可作为物联网节点&#xff0c;实现物联网应用。同时掌控板上集成了OLED…

【力扣】链表题目总结

文章目录 链表基础题型一、单链表翻转、反转、旋转1.反转链表2.反转链表II——反转部分链表3.旋转链表4.K个一组翻转链表5.反转偶数长度组的节点 二、删除单链表中的结点1.删除链表的结点2.删除未排序链表中的重复节点3.删除已排序链表中的重复元素I——重复元素只剩下一个4.删…

分库分表概念、原理、拆分策略和实现技术讲解

文章目录 1.什么是分库分表2.分库分表拆分策略2.1 垂直拆分2.2 水平拆分 3.分库分表实现技术简介 1.什么是分库分表 分库分表的中心思想就是将数据分散存储&#xff0c;使得单一数据库/表的数据量变小来缓解单一数据库的性能问题&#xff0c;从而达到提升数据库性能的目的。 …

Redis面试题2

Redis面试题-2 10、统计高并发网站每个网页每天的 UV 数据&#xff0c;结合Redis你会如何实现&#xff1f; 选用方案&#xff1a;HyperLogLog 如果统计 PV 那非常好办&#xff0c;给每个网页一个独立的 Redis 计数器就可以了&#xff0c;这个计数器的 key 后缀加上当天的日期…

vue+element中如何设置单个el-date-picker开始时间和结束时间关联

功能&#xff1a;选了开始时间&#xff0c;则结束时间只能选择开始时间之后的&#xff1b;选了结束时间&#xff0c;则开始时间只能选择结束时间之前的 重点是picker-options属性 图示&#xff1a; 代码展示: // body 内部<el-form-item><el-date-pickerv-model&qu…

提高测试用例质量的6大注意事项

在软件测试中&#xff0c;经常会遇到测试用例设计不完整&#xff0c;用例没有完全覆盖需求等问题&#xff0c;这样往往容易造成测试工作效率低下&#xff0c;不能及时发现项目问题&#xff0c;无形中增加了项目风险。 因此提高测试用例质量&#xff0c;就显得尤为重要。一般来说…

day50-springboot+ajax分页

分页依赖&#xff1a; <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper-spring-boot-starter</artifactId> <version>1.0.0</version> </dependency> 配置&#xff1a; …

Linux中提示No such file or directory解决方法

说明&#xff1a; 在linux下&#xff0c;./xxx.sh执行shell脚本时会提示No such file or directory。但shell明明存在&#xff0c;为什么就是会提示这个呢&#xff1f; 这种其实是因为编码方式不对&#xff0c;如你在win下编辑sh&#xff0c;然后直接复制到linux下面 实现&…

记一次ubuntu16误删libc.so.6操作的恢复过程

背景 操作系统&#xff1a;ubuntu16 glibc版本&#xff1a;2.23 修改原因&#xff1a; 经过一系列报错和手工构建之后&#xff0c;vulkansdk成功安装&#xff08;起码运行./vulkansdu成功&#xff09;&#xff0c;在进行./vulkaninfo进行验证时&#xff0c;报错&#xff1a…

Redis 搭建主从集群

文章目录 1. 主从集群架构1.1 准备实例和配置1.2 启动1.3 开启主从关系1.4 测试 2. 主从同步原理2.1 全量同步2.2 增量同步repl_backlog原理 2.3 主从同步优化小结 单节点的 Redis 并发能力有限&#xff0c;要进一步提高 Redis 的并发能力&#xff0c;就需要搭建主从集群&#…

奥威BI系统:零编程建模、开发报表,提升决策速度

奥威BI是一款非常实用的、易用、高效的商业智能工具&#xff0c;可以帮助企业快速获取数据、分析数据、展示数据。值得特别注意的一点是奥威BI系统支持零编程建模、开发报表&#xff0c;是一款人人都能用的大数据分析系统&#xff0c;有助于全面提升企业的数据分析挖掘效率&…

LabVIEW使用DSA技术从X射线图像测量肺气容量

LabVIEW使用DSA技术从X射线图像测量肺气容量 相衬X射线&#xff08;PCX&#xff09;成像技术利用相邻介质之间折射率的微小差异来增强传统X射线成像通常不可见的物体的边界。事实证明&#xff0c;这一进展在一系列生物医学和材料科学中非常有益于材料表征、疾病检测以及解剖形…

springboot整合tio-websocket方案实现简易聊天

写在最前&#xff1a; 常用的http协议是无状态的&#xff0c;且不能主动响应到客户端。最初想实现状态动态跟踪只能用轮询或者其他效率低下的方式&#xff0c;所以引入了websocket协议&#xff0c;允许服务端主动向客户端推送数据。在WebSocket API中&#xff0c;浏览器和服务…

ThinkPHP 6 添加跳转提示扩展 liliuwei/thinkphp-jump

liliuwei/thinkphp-jump 是 TP5 中经典跳转提示&#xff0c;在 TP6 中已经取消&#xff0c;通过 composer 下载该扩展可以在 TP6 中使用 TP5 的跳转提示操作。 安装扩展 在应用根目录执行: composer require liliuwei/thinkphp-jump引入扩展 在全局配置目录生成 jump.php 文件…

Eeny Meeny Moo

Eeny Meeny Moo 题目描述输入输出格式输入格式输出格式 输入输出样例输入样例输出样例 正确解法A C 代码 题目描述 你肯定有过这样的经验&#xff0c;那就是当很多一起使用网络的时候&#xff0c;网速变得很慢很慢。为了解决这个问题&#xff0c;德国的Ulm大学开发了一份意外事…

【GEMM预备工作】行主序和列主序矩阵的内存中的连续性,解决理解问题

在内存存储中&#xff0c;默认矩阵是按照行优先储存的&#xff0c;即矩阵的每一列在内存中是连续的。行优先矩阵储存中行数据是不连续的。 而对于列主序的矩阵&#xff0c;是按照列优先储存的&#xff0c;即矩阵的每一行在内存中是连续的。列优先矩阵储存中列数据是不连续的&am…

命令模式(Command)

命令模式是一种行为设计模式&#xff0c;可将一个请求封装为一个对象&#xff0c;用不同的请求将方法参数化&#xff0c;从而实现延迟请求执行或将其放入队列中或记录请求日志&#xff0c;以及支持可撤销操作。其别名为动作(Action)模式或事务(Transaction)模式。 Command is …