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

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将父类中的析构函数定义为纯虚析构函数

子类中的析构函数被调用,将父类中的析构函数声明为纯虚析构函数后,必须要实现声明的函数。
在这里插入图片描述

http://www.lryc.cn/news/26672.html

相关文章:

  • Android性能优化系列篇:弱网优化
  • Mysql 插入大批量数据调优方法
  • matlab基础
  • 自动化测试——多窗口切换和切换frame
  • C#中,Elasticsearch.Net判断空字符串
  • 23种设计模式-适配器模式
  • 深入理解this指向问题
  • 事业单位联考(综合应用A类)典型例题教案
  • frp内网穿透实验
  • 认识JavaScript中的防抖函数
  • macOS 13.3 Beta 2 (22E5230e)With OpenCore 0.8.9正式版 and winPE双引导分区原版镜像
  • JetPack—DataStore核心原理与使用
  • 热烈祝贺|酒事有鲤盛装亮相2023中国(山东)精酿啤酒产业发展创新论坛暨展览会
  • 深度强化学习DLR
  • Android Handler机制(四) Message源码分析
  • 【Git】git命令(全)
  • 软考论文-成本管理(1)
  • Java 多线程 --- 锁的概念和类型划分
  • python程序员狂飙上头——京海市大嫂单推人做个日历不过分吧?
  • 浅谈子网掩码、IP地址、网络地址之间关系
  • 前端优化的解决方案
  • PYthon组合数据类型的简单使用
  • 【Java】P2 基础语法与运算符
  • 【并发基础】Java中线程的创建和运行以及相关源码分析
  • Spark Shuffle
  • Linux/MacOS 生成双击可执行文件
  • Ubuntu三种拨号方法
  • Vue-router的引入和安装
  • 无线WiFi安全渗透与攻防(四)之kismet的使用
  • 2023新版PMP考试有哪些变化?