多态——细致讲解

🔶多态基础概念
 🔶概念
  🔱多态性
  🔱多态——重新(覆盖)
 🔶示例
  🔶基本使用方法
  🔶特例
   🔱协变
   🔱析构函数重写
 🔱多态原理
  🔱1. 虚函数形成虚表
  🔱2. 虚函数存储位置(覆盖)
  🔱3. 多态中重写的虚函数存储位置
   🔱1. 重写原理——虚表
   🔱2. 单继承中,子类新增虚函数会存到父类的虚表中——普通继承
   🔱3. 单继承中,子类新增虚函数会存到父类的虚表中——虚继承
   🔱4. 同类公用一个虚表;父类和子类不共用一张虚表
 🔱多态例题
🔱经典问题

多态基础概念

概念

多态性

 1. 静态多态:函数重载和运算符重载
 2. 动态多态:继承和虚函数

多态——重写(覆盖)

 1. 父类的指针/引用调用虚函数
 2. 调用的虚函数必须是子类重写的虚函数
这样就能在指针调用相应的对象函数的时候使用相应的成员函数,具体看示例
这里条件很严格
重写的函数要是一摸一样——返回值,函数名,参数个数,参数位置,参数类型都要完全一样,虚函数之后的const也要一样

示例

基本使用方法
  1. 父类中需要使用virtual修饰函数,子类中virtual可以不写
class A
{
public:virtual void func(){puts("A-->func");}
};
class B:public A
{
public:virtual void func(){puts("B-->func");}
};
int main()
{// 父类指向子类A* a1 = new B;a1->func();// 父类指向父类a1 = new A;a1->func();// 父类引用子类B tb;A& a2 = tb;a2.func();// 父类引用父类A ta;A& a3 = ta; // 不能直接使用a2=ta,引用不能重新赋值,虽然他不会报错,但是他的结果是错的a3.func();return 0;
}

在这里插入图片描述


  1. final 修饰类——不能继承

在这里插入图片描述

修饰虚函数——不能背重写

在这里插入图片描述

  1. override ——这个函数一定要重新父类的某一个虚函数

在这里插入图片描述

一定要注意这两个关键字加载虚函数结尾

特例
协变

虚函数的返回值可以不一样,只能出现父类返回父类的指针/引用,子类返回子类的指针/引用

在这里插入图片描述

不可以一个返回指针,一个返回引用
只能同时返回指针/同时返回引用

析构函数重写
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}~A(){cout << "delete A" << endl;}int _a = 1;};
class B :public A
{
public:virtual void fun1(){cout << "B::fun1" << endl;}virtual void fun3(){cout << "B::fun3" << endl;}~B(){cout << "delete B" << endl;}int _b = 2;
};
int main()
{A* a = new B;delete a;return 0;
}

在这里插入图片描述

delete释放看的是类型,也就是说这里delete调用的是A的析构函数
根本上说,delete会被处理成—> destructor() + operator delete,所以他们能构成重写,在具体实现的时候需要写成virtual

	virtual ~A(){cout << "delete A" << endl;}

在这里插入图片描述


多态原理

1. 虚函数形成虚表
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}};
int main()
{A a;return 0;
}

在这里插入图片描述

2. 虚函数存储位置

虚函数和普通函数放在一起,虚表存储在代码段

3. 多态中重写的虚函数存储位置
🎭1. 重写原理——虚表
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}};
class B :public A
{
public:virtual void fun1(){cout << "B::fun1" << endl;}
};
int main()
{B b;return 0;
}

在这里插入图片描述

🎭2. 单继承中,子类新增虚函数会存到父类的虚表中——普通继承
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}};
class B :public A
{
public:virtual void fun1(){cout << "B::fun1" << endl;}virtual void fun3(){cout << "B::fun3" << endl;}
};
int main()
{B b;print((T*)(*(int*)(&b)));return 0;
}

在这里插入图片描述
vs中虚表通常在最后一个都是0,Linux不是

在这里插入图片描述

🎭3. 单继承中,子类新增虚函数会存到父类的虚表中——虚继承
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}int _a = 1;};
class B :virtual public A
{
public:virtual void fun1(){cout << "B::fun1" << endl;}virtual void fun3(){cout << "B::fun3" << endl;}int _b = 2;
};
int main()
{B b;A* a = &b; // 要注意这种写法,确保他能准确跳到下一个虚表处print((T*)(*(int*)(a)));print((T*)(*(int*)(&b)));return 0;
}

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

🎭4. 同类公用一个虚表;父类和子类不共用一张虚表
#include<iostream>
using namespace std;
typedef void (*T)();
class A
{
public:virtual void fun1(){cout << "A::fun1" << endl;}virtual void fun2(){cout << "A::fun2" << endl;}int _a = 1;};
class B :virtual public A
{
public:virtual void fun1(){cout << "B::fun1" << endl;}int _b = 2;
};
int main()
{A a;B b1;B b2;return 0;
}

在这里插入图片描述


🖼多态例题

class A
{
public:virtual void fun(int val = 1){cout << "val = " << val << endl;}virtual void test(){fun();}
};
class B :public A
{
public:virtual void fun(int val = 0){cout << "val = " << val << endl;}
};
int main()
{A* a = new B;a->test();return 0;
}
void print(T a[])
{for (int i = 0; a[i] != 0; i++){printf("[%d]--->%p\n", i, a[i]);}puts("");
}
int main()
{B b;print((T*)(*(int*)(&b)));return 0;
}

在这里插入图片描述

  1. 父类指向子类,调用的test函数,test函数是父类的虚函数,类内的函数有一个默认的this指针,test内部调用的fun函数实际上是this->fun(this类型是A*——父类的指针指向虚函数),fun是子类重写的虚函数(函数是子类重写的虚函数)——满足多态条件
  2. 虚函数中的this是根据是否重写确定的,这里的test没有被重写,是A*this指针,然后调用fun,fun是经过重写的函数,所以调用的是重写的函数
  3. 虚函数继承的是函数的接口,重写的是函数的实现

所以缺省值才是1


#include<iostream>
using namespace std;class A
{
public:virtual void fun(int val = 0){printf("A::fun()--> %d", val);}virtual void test(){fun();}
};
class B:public A
{
public:void fun(int val = 1){printf("B::fun()--> %d", val);}
};
int main()
{B b;b.test();return 0;
}

在这里插入图片描述


🔒经典问题

  1. 什么是多态?
  2. 什么是重载、重写(覆盖)、重定义(隐藏)?
  3. 多态的实现原理?
  4. inline函数可以是虚函数吗?

可以,不构成多态就是inline,构成多态就不是inline

  1. 静态成员可以是虚函数吗?

不能,因为静态成员函数没有this指针,使用类型::成员函数
的调用方式无法访问虚函数表,所以静态成员函数无法放进虚函数表。

  1. 构造函数可以是虚函数吗?

不能,虚表在编译时生成
在调用构造函数之后,但是虚表指针在成员初始化之前

  1. 析构函数可以是虚函数吗?

本就应该是,在A* a = new B;这种场景下,在释放子类对象时,需要将析构函数变成虚函数

  1. 对象访问普通函数快还是虚函数更快?

首先如果是普通对象,是一样快的。如果构成多态,就是普通函数快,因为运行时调用虚函数需要到虚函数表中去查找。

  1. 虚函数表是在什么阶段生成的,存在哪的?

虚函数表是在编译阶段就生成的,一般情况下存在代码段(常量区)的。

  1. C++菱形继承的问题?虚继承的原理?

菱形继承会造成祖宗类数据冗余的问题
在每一个继承自祖宗类的派生类中,使用一个指针指向一个偏移量,根据偏移量找到的地址就是祖宗类的数据,并且这个数据只有一份

  1. 什么是抽象类?抽象类的作用?

抽象类含有形如 virtual void fun() =0; 的基类/派生类
强制派生类重写父类的实现


原理的角度理解,重写之后将fun虚函数进行覆盖test是A*this调用经过重写的虚函数fun符合多态的条件,并且继承的是接口不是实现fun虚函数继承父类函数接口,并使用重写的虚函数实现,最终形成了这个样子


优秀多态文章
优秀多态文章
为什么要使用父类指针和引用实现多态,而不能使用对象?
虚析构函数
虚表位置
虚表位置

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

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

相关文章

LaTeX-设置表格大小

文章目录 LaTeX-设置表格大小1.创建表格2.设置表格的宽度2.1控制表格每一列的宽度2.2控制整个表格的宽度 3.设置表格的外观4.LaTeX绘制三线表 LaTeX-设置表格大小 本文介绍了LaTeX如何设置表格的大小、改变表格的外观以及如何绘制三线表。 1.创建表格 在LaTeX中创建表很耗时…

算法修炼-动态规划之路径问题(1)

62. 不同路径 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a;选定一个网格为终点&#xff0c;走到这个网格的所有走法就是这个网格的上面一个网格的所有走法加上这个网格左边一个网格的所有走法&#xff0c;然后做好初始化工作就行。 class Solution { public:int…

扼杀网络中的环路:STP、RSTP、MSTP

目录 前言&#xff1a; 一、STP&#xff08;Spanning Tree Protocol&#xff09; 1.1 STP功能 1.2 STP应用 二、RSTP&#xff08;Rapid Spanning Tree Protocol&#xff09; 2.1 RSTP功能 2.2 RSTP应用 三、MSTP&#xff08;Multiple Spanning Tree Protocol&#xff0…

【中英对照】【自译】【精华】麻省理工学院MIT技术双月刊(Bimonthly MIT Technology Review)2024年3/4月刊内容概览

一、说明 Notation 仅供学习、参考&#xff0c;请勿用于商业行为。 二、本期封面、封底 Covers 本期杂志购于新加坡樟宜机场Changi Airport Singapore&#xff0c;售价为20.50新元。 本期仍然关注伦敦的AI大会。&#xff08;笔者十分想去&#xff0c;在伦敦和MIT校园均设有会…

SpringBoot整合rabbitmq-扇形交换机队列(三)

说明&#xff1a;本文章主要是Fanout 扇形交换机的使用&#xff0c;它路由键的概念&#xff0c;绑定了页无用&#xff0c;这个交换机在接收到消息后&#xff0c;会直接转发到绑定到它上面的所有队列。 大白话&#xff1a;广播模式&#xff0c;交换机会把消息发给绑定它的所有队…

进程间的通信 -- 共享内存

一 共享内存的概念 1. 1 共享内存的原理 之前我们学过管道通信&#xff0c;分为匿名管道和命名管道&#xff0c;匿名管道通过父子进程的属性继承原理来完成父子进程看到同一份资源的目的&#xff0c;而命名管道则是通过路径与文件名来唯一标识管道文件&#xff0c;来让不同的进…

【系统分析师】-需求工程

一、需求工程 需求工程分为需求开发和需求管理。 需求开发&#xff1a;需求获取&#xff0c;需求分析&#xff0c;需求定义、需求验证。 需求管理&#xff1a;变更控制、版本控制、需求跟踪&#xff0c;需求状态跟踪。&#xff08;对需求基线的管理&#xff09; 1.1需求获取…

02-设计概述

上一篇&#xff1a;01-导言 本章重点讨论 JNI 中的主要设计问题。本节中的大多数设计问题都与本地方法有关。调用 API 的设计将在第 5 章&#xff1a;调用 API 中介绍。 2.1 JNI 接口函数和指针 本地代码通过调用 JNI 函数来访问 Java 虚拟机功能。JNI 函数可通过接口指针使用…

如何实现桌面美化

一.隐藏桌面图标 1. 在商店里下载TranslucentTB 二.设置底层图标 1.下载Nexus 打开官网&#xff1a; Winstep Nexus Dock and Nexus Ultimate - The Advanced Docking System for Windows 三.设置插件 1.打开致美化官网 致美化 - 最专业的视觉美化交流平台 (zhutix.com) 2.注…

MySQL进阶:全局锁、表级锁、行级锁总结

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位大四、研0学生&#xff0c;正在努力准备大四暑假的实习 &#x1f30c;上期文章&#xff1a;MySQL进阶&#xff1a;MySQL事务、并发事务问题及隔离级别 &#x1f4da;订阅专栏&#xff1a;MySQL进阶 希望文章对你们有所帮助…

swagger在java中的基本使用

自动生成接口文档&#xff0c;和在线接口测试的框架。 导入依赖 <!-- knife4j对swagger进行一个封装--><dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-spring-boot-starter</artifactId><versi…

LeetCode 刷题 [C++] 第121题.买卖股票的最佳时机

题目描述 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票&#xff0c;并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利润。 返回你可以从这笔交易中获取的…