【C++进阶02】多态

在这里插入图片描述

一、多态的概念及定义

1.1 多态的概念

多态简单来说就是多种形态
同一个行为,不同对象去完成时
会产生出不同的状态
多态分为静态多态动态多态
静态多态指的是编译时
在程序编译期间确定了程序的行为
比如:函数重载
动态多态指的是运行时
在程序运行期间,根据具体拿到的类型
确定程序的具体行为,调用具体的函数

1.2 在继承中要构成多态的两个条件

  1. 必须通过父类指针或引用调用虚函数

  2. 虚函数的重写
    函数名、参数类型、返回值都要相同

被virtual修饰的类成员函数称为虚函数

class Person {
public:virtual void BuyTicket() { cout << "买票-全价" << endl;}
};

1.3 虚函数的重写(覆盖)

派生类中有一个跟基类完全相同的虚函数
(即派生类虚函数与基类虚函数的返回值类
型、函数名字、参数列表完全相同)
称子类的虚函数重写了基类的虚函数

普通函数的继承是实现继承
派生类继承了基类函数,可以使用函数
继承的是函数的实现
虚函数的继承是接口继承
派生类继承的是基类虚函数的接口
目的是为了重写,达成多态,继承的是接口
如果不实现多态,不要把函数定义成虚函数

class Person {
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }// 只要父类析构加了virtual就构成多态,子类加不加都可以正确释放virtual ~Person() { cout << "~Person" << endl; };
};class Student : public Person {
public: // 子类可以不写virtual,因为他继承父类的接口,重写实现virtual void BuyTicket() { cout << "买票-半价" << endl; }~Student() { cout << "~Student" << endl; }
};void Func(Person& p)
{ p.BuyTicket(); }int main()
{
Person ps;
Student st;// 构成多态后
Func(ps); // 传父类调用父类的虚函数
Func(st); // 传子类调用子类的虚函数return 0;
}

1.4 协变

如果是父子关系的指针或引用
返回值可以不同也构成多态

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;}
};

1.5 final和override

final: 修饰虚函数
表示该虚函数不能再被重写
现实中不常用,不能实现多态的虚函数
意义不大

class Car
{
public:virtual void Drive() final {}
};
class Benz :public Car
{
public:virtual void Drive() {cout << "Benz-舒适" << endl;}
};

override: 检查派生类虚函数
是否重写了基类某个虚函数
如果没有重写编译报错

class Car{
public:virtual void Drive(){}
};
class Benz :public Car {
public:virtual void Drive() override {cout << "Benz-舒适" << endl;}
};

1.6 重载、覆盖(重写)、隐藏(重定义)的对比

面试题经常被问到
在这里插入图片描述

1.7 抽象类

在虚函数后面加上 =0
这个函数就叫纯虚函数
包含纯虚函数的类叫做抽象类
抽象类不能实例化出对象
派生类继承后也不能实例化出对象
只有重写纯虚函数,派生类才能实例化出对象

class Car
{
public:// 纯虚函数 --- 抽象类virtual void Drive() = 0;
};int main()
{Car car; // 无法实例化对象return 0;
}

二、多态的原理

2.1 虚函数表

这里常考一道笔试题:sizeof(Base)是多少?

class Base
{
public:virtual void Func1(){cout << "Func1()" << endl;}
private:int _b = 1;
};int main()
{cout << sizeof(Base) << endl;return 0;
}

在32位操作系统下是8 bit
在64位操作系统下是16 bit
通过调试发现还有个指针_vfptr
这个指针叫做虚函数表指针
本质是指针数组
用来存放虚函数的地址
在这里插入图片描述

对象中存的是虚表指针
虚表存的是虚函数指针
虚函数和普通函数一样的
都是存在代码段的

2.2 多态的原理

通过下面代码观察父子类
的虚表之间的关系

class Base
{
public:virtual void Func(){cout << "Base::Func1()" << endl;}
};
class Derive : public Base
{
public:virtual void Func(){cout << "Derive::Func1()" << endl;}
};void Test(Base* p)
{p->Func();
}int main()
{Base b;Derive d;Test(&b);return 0;
}

监视窗口
在这里插入图片描述
通过监视窗口可以发现
派生类对象d中也有一个虚表指针
通过地址发现基类和派生类的虚表是不一样的

虚函数表本质是存虚函数指针的指针数组
一般情况这个数组最后面放了一个nullptr

结论:
观察下图红色箭头
当传过来的是父类对象的地址
p->Func在父类的虚表中找对应的虚函数Func地址
当传过来的是子类对象的地址
p->Func在子类的虚表中找对应的虚函数Func地址

这样就实现不同对象的同一行为
展现的不同状态
在这里插入图片描述
当父类有虚函数而子类没有虚函数
也没有重名函数
子类是不能继承父类的虚表指针
子类会生成一个虚表指针
存父类的虚函数地址
当子类有虚函数而父类没有
父类也就不会有虚表
因为子类有的东西父类不一定有

派生类的虚表生成:
a.先将基类中的虚表内容拷贝一份到派生类虚表中
b.如果派生类重写了基类中某个虚函数
用派生类自己的虚函数覆盖虚表中基类的虚函数
c.派生类自己新增加的虚函数按其在派生类中的
声明次序增加到派生类虚表的最后

2.3 在多态下建议把基类的析构函数定义成虚函数

如果基类析构函数不是虚函数
就调不到派生类的析构函数
(指针类型是父类,所以调用父类的析构函数)
从而造成内存泄漏

class Person {
public:virtual ~Person() {cout << "~Person()" << endl;}
};
class Student : public Person {
public:virtual ~Student() { cout << "~Student()" << endl; }
};int main()
{Person* ps = new Student;delete ps;return 0;
}

形成多态的条件之一便是
只能通过父类去调用
所以子类对象只能强转成父类类型
如果父类的析构函数不是虚函数
那子类便调不到自己的析构函数
因为子类对象的类型是父类
所以只能调用父类的析构函数
子类成员无法释放从而造成内存泄漏
父类析构函数定义成虚函数便能解决问题

✨✨✨✨✨✨✨✨
本篇博客完,感谢阅读🌹
如有错误之处可评论指出
博主会耐心听取每条意见
✨✨✨✨✨✨✨✨

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

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

相关文章

变限积分求导(带参,极限)

方法 一般形 带参数方程形 带极限型

IDEA 中 Tomcat 日志乱码

1、服务器输出乱码 修改 File -> settings -> Editor -> General ->Console 中&#xff0c;utf-8改为GBK&#xff0c;反之改成utf-8 2、Tomcat Localhost Log 或者 Tomcat Catalina Log乱码 进入Tomcat 中的conf文件中的logging.properties 哪个有问题改哪个&…

Ubuntu:VS Code上C++的环境配置

使用 VSCode 开发 C/C 程序 , 涉及到 工作区的.vscode文件夹下的3个配置文件&#xff08;均可以手动创建&#xff09; : ① tasks.json : 编译器构建 配置文件 ; ② launch.json : 调试器设置 配置文件 ; ③ c_cpp_properties.json : 编译器路径和智能代码提示 配置文件 ; …

竞赛保研 基于GRU的 电影评论情感分析 - python 深度学习 情感分类

文章目录 1 前言1.1 项目介绍 2 情感分类介绍3 数据集4 实现4.1 数据预处理4.2 构建网络4.3 训练模型4.4 模型评估4.5 模型预测 5 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于GRU的 电影评论情感分析 该项目较为新颖&#xff0c;适合作为竞…

uniapp怎么动态渲染导航栏的title?

直接在接口请求里面写入以下&#xff1a; 自己要什么参数就写什么参数 本人仅供参考&#xff1a; this.name res.data.data[i].name; console.log(名字, res.data.data[i].name); uni.setNavigationBarTitle({title: this.name}) 效果&#xff1a;

界面控件DevExpress v23.2全新发布 - 官宣正式支持.NET 8

DevExpress拥有.NET开发需要的所有平台控件&#xff0c;包含600多个UI控件、报表平台、DevExpress Dashboard eXpressApp 框架、适用于 Visual Studio的CodeRush等一系列辅助工具。屡获大奖的软件开发平台DevExpress 今年第一个重要版本v23.1正式发布&#xff0c;该版本拥有众多…

计算机网络(5):运输层

这一章应该是整个计算机网络对我们来说最重要的&#xff0c;也是用的最多的一部分。 运输层协议 进程之间的通信 从通信和信息处理的角度看&#xff0c;运输层向它上面的应用层提供通信服务&#xff0c;它属于面向通信部分的最高层&#xff0c;同时也是用户功能中的最低层。…

c++ 使用 at()访问数组 抛出异常

1、说明 当我们定义一个数组vector b(10)后&#xff0c;b[]和b.at()都可以对v中元素进行访问&#xff0c;平时一般大家使用的都是v[]这种访问方法&#xff0c;以至于将v.at()这种访问方式忘记了。 2、vector[]和vector.at()的区别 b.v[]和b.at()都可以对v中元素进行访问&…

12.23C语言 指针

& 地址运算符&#xff0c;用于取地址 /*注释内容*/ //注释一行 *的意思&#xff1a;1.算术运算符 2.用于指针声明int *ptr;表示这个变量是一个指针3.数组元素访问&#xff1a;在数组名后面使用 * 可以表示数组的起始地址。例如&#xff1a; int arr[5] {1, 2, 3, 4, 5…

【5G PHY】NR参考信号功率和小区总传输功率的计算

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…

超分辨数据集:Set5 Set14 BSD100 Urban100 Manga109

DIV2K数据集官网上很好找到&#xff0c;但是网上流传的Set5 14 BSD100,Urban100 Manga109都是私人进行处理过的版本&#xff0c;各个处理方式都不同&#xff0c;为了统一方式写了这篇文章。 官方的DIV2K x2、x3、x4的LR图片使用下面matlab代码生成&#xff08;已经经过测试最后…

单例模式实现

⭐ 作者&#xff1a;小胡_不糊涂 &#x1f331; 作者主页&#xff1a;小胡_不糊涂的个人主页 &#x1f4c0; 收录专栏&#xff1a;JavaEE &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 单例模式 1. 什么是单例模式2. 饿汉模式3.…