yo!这里是c++中的多态

前言

        在学完继承之后,紧接着我们来认识多态,建议继承不太熟的先把继承部分的知识点搞熟,再来学习多态,否则会走火入魔,会混乱。因为多态是建立在继承的基础之上,而且多态中还存在与继承类似的概念,所以学习多态的过程中多多与继承区别,做些对比会容易区分并且记住。

        对于多态,通俗点说就是多种形态,具体来说就是不同的对象去完成某个行为时,产生出不同的状态。举两个例子,比如买票这个行为,成人买是全票,学生是半价买票,军人是优先买票;再比如,支付宝抢红包,老用户就少些,新用户就多些,这些都是一种多态行为,具体向下看看实现和原理等相关知识点吧。

多态介绍

  • 构成条件

①通过基类的指针或引用调用虚函数;

②被调用的函数是虚函数,且派生类必须重写基类的虚函数,

        如下图,通过Func函数进而使用引用去调用虚函数,在Person-Student的父子类中,子类重写了父类的虚函数BuyTicket,由此构成了多态,满足了不同的人购买票的条件不一样。

eg: 

  • 虚函数重写

虚函数:被virtual修饰的类成员函数。

eg: 

重写:也叫覆盖,派生类的一个虚函数与基类的一个虚函数的函数名、返回值类型、参数列表完全相同,叫做派生类的虚函数重写了基类的虚函数。

eg:

        以上就是构成多态的两个条件,除此之外还有两个例外,这两个例外也构成多态。

注意:

        ①除了两个例外,不符合重写就是符合隐藏关系,

        ②子类的的虚函数不加virtual关键字也可以,但建议加上。

  • 两个例外

1.协变

        基类和派生类的虚函数返回值可以不同,但必须分别是具有父子关系的类的指针或引用,这样也可以构成多态。当然,也不局限于当前基类和派生类,其他具有父子关系的也可以,但一定要父类返回父类,子类返回子类。

eg:

2.析构函数的重写

        如果基类的析构函数处理成虚函数,无论派生类的析构函数加不加virtual,那么基类和派生类的析构函数构成重写,进而构成多态。

        一是多态关系中,即使派生类中的函数不加virtual关键字,也依旧构成多态;

        二是编译器会对每个类的析构函数做处理,编译后析构函数的名会被统一处理成destructor,所以满足了同名的条件,

        存在这个特例的主要目的就是为析构函数构成多态做铺垫,所以建议在继承中析构函数定义成虚函数,那为什么要让析构函数构成多态呢?看下面这种情况:

        通过new的方式实例化对象,因为是new了Person和Student类的空间,所以希望delete可以将这两块空间成功释放掉,但是如果是下面这种类的实现,就会产生问题

        我们发现,delete p2调用的是Person的析构函数,new出来的Student空间没有完全释放掉,为什么?因为这是普通调用,并没构成多态,delete p2(①p2->destructor;②operator delete(p2);)中,在编译时期就确定了destructor的函数地址,再看到p2是Person类型的指针,所以调用Person的destructor,导致意想不到的错误。如何处理?见下方类实现:

        只有派生类的析构函数重写基类的析构函数,才能构成多态,进而保证p1、p2指向的对象正确调用析构函数。

        注意上面这种特例不要与下面两种普通情况搞混,下面两种无需构成多态也能正确调用析构函数:

        一般情况下,大家只需要记住这种特例就行,或者习惯在具有继承关系的两个类的析构函数加上virtual关键字,进而构成多态,否则在需要构成多态的情境下会出现意想不到的结果。

  • override和final

        overdide和final这是两个关键字,由c++11提供,可以帮助用户检测虚函数是否被重写,因为重写的条件比较多,有时候会因为疏忽(比如函数名并没有相同)并没有构成多态,但编译器不会报错,所以有时候会加大debug的难度。

final:表示该函数不能被重写

注意:

        ①使用场景极少;

        ②final还可用修饰类,表示禁止其他类继承,比如class DontDerive final{};。

eg:

override:一般修饰子类的虚函数,检测此虚函数是否重写了父类的某个虚函数。

eg:

抽象类

纯虚函数:在虚函数后面加上【= 0】。

抽象类:包含纯虚函数的类,也叫接口类

        抽象类不能实例化出对象,派生类继承抽象类后也不能实例化出对象,只有重写纯虚函数,才能实例化出对象。这属于一种间接强制重写的手段,而且纯虚函数的重写体现出了接口继承,即派生类继承的是基类虚函数的接口(函数头),使用自己的函数体,也就是基类的虚函数的出现本质就是给派生类虚函数重写的。与接口继承对应的就是实现继承,普通函数的继承就是实现继承,基类实现的函数,派生类继承以后可以使用。

eg:

多态原理

        如下图,Person对象中有一个虚函数和一个属性sno,通过监视我们可以看到,Person类实例化出的对象中除了sno外还有一个指针变量__vfptr,实际上,这个指针是虚函数表指针(简称虚表指针),指向一个虚函数表(简称虚表),而在虚函数表中就存在着类中的虚函数地址。值得注意的是,一个含有虚函数的类中都至少有一个虚表指针,且指向一个虚表

eg:

        到底多态的原理是什么呢?每个包含虚函数的类都会有一个虚函数表,虚函数表是一个存放着虚函数地址的指针数组,其中每个指针指向该类的虚函数。当进行函数调用时,通过对象的虚函数指针找到对应的虚函数表,然后根据函数在虚函数表中的索引找到实际要调用的函数,这个过程不是在编译时发生的,而是在运行时发生的;不满足多态的函数调用则是在编译时就确认函数地址然后运行时直接调用。

单继承和多继承的虚表 

  • 单继承的虚表

        如下图,可以看到父类Person的虚表中有func2和func3,func1不是虚函数,所以不存在虚表中,而在子类Student的虚表之中,我们可以看到,不仅存在父类中的虚函数,且重写了对应虚函数,但是并没有看到子类自己的虚函数,其实有的,只是编译器的监视窗口并未显示,下面会用代码打印出两个虚表的虚函数,具体看一下两个类中虚函数情况。不过,在此之前先总结一下子类的虚表生成过程:

        ①先将基类中的虚表内容拷贝一份到派生类虚表中;

        ②如果派生类重写了基类中某个虚函数,用派生类自己的虚函数覆盖虚表中基类的虚函数;

        ③派生类自己的虚函数按其在派生类中的声明次序增加到派生类虚表的最后。

注意:虚函数并不存在虚表中,虚表也不存在类和对象中。而是,虚表中存在着虚函数的地址,虚函数和普通函数一样,存在于代码段中,只是地址存在了虚表中而已,并且对象中存的是虚表指针,不是虚表,而虚表也是存在于代码段之中的(不同平台结果不一样)。

eg:

 打印类的虚函数的代码:

        VFPTR是被typedef的一个函数指针,函数传入一个虚表指针,遍历虚表中的虚函数地址,分别调用一下对应虚函数。虚表本质上就是一个存放虚函数地址的指针数组,最后会有一个nullptr,所以调用到空指针为止。

        那如何使用这个函数呢,也就是如何取到一个类的虚表指针呢?

        在vs下,虚表指针总是在一个类的开头,32位下指针大小为4个字节,所以取到前4个字节就能拿到虚表指针,那如何取到对象的前4个字节呢?

①先取对象p的地址,强转成一个int*的指针;

②再解引用就得到了前4个字节;

③将这4个字节强转成VFPTR*,因为函数需要传入一个VFPTR*。

        看到运行结果,在子类的虚表中,除了继承的父类的虚函数,还有自己的虚函数。

typedef void(*VFPTR)();
void PrintVFTable(VFPTR* table)
{for (size_t i = 0; table[i] != nullptr; ++i){printf("vft[%d]:%p->", i, table[i]);VFPTR pf = table[i];pf();}cout << endl;
}

运行:

  • 多继承的虚表

        多继承的情况与单继承差不多,就是派生类多了一个或多个基类之后,派生类对象的虚函数地址是放在哪一个基类的虚表中,看下面的例子:

        我们发现,派生类的虚函数重写了所有父类的虚函数,则所有父类的虚函数都会被覆盖,但是在监视窗口中并没有发现派生类的虚函数地址放在了哪个基类的虚表中,运行打印类的虚函数的代码试试看:

        观察上图发现:在多继承中派生类的虚函数会放在第一个继承基类的虚表中。结论很号看出,但是C类中的B类的前四个字节如何取出的呢?

        方法一:在C类中,先继承的基类肯定先放在前面,所以先取地址c强转成char类型指针(因为后面要加上A大小的字节,所以不能直接强转成int类型的指针),再跨过A大小个字节,强转为int指针类型,然后再强转成VFPTR*,防止类型不匹配。

        方法二:利用切片取到C中B的部分,再按照之前的方法取到前4个字节,相比之下,方法二更容易理解。

面试常见问题

1.inline函数可以是虚函数吗?

        内联函数不可以是虚函数,或者说内联函数与虚函数本质上是相斥的,因为内联函数直接在代码中展开,无函数地址,而作为虚函数,函数地址将会存进虚表中。但作为虚函数时再加上inline关键字,编译也不会报错,因为inline对编译器是一种建议,编译器会自动忽略inline。

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

        不可以,因为静态成员函数没有this指针,虚函数地址存进虚表中,需要this指针取到虚函数地址进而调用虚函数。

3.构造函数、拷贝构造函数、operator=可以是虚函数吗?

        构造函数不可以是虚函数,因为对象中的虚表指针是在构造函数中初始化的(可用监视窗口验证),调用构造函数时还找不到虚表来存储构造函数的地址,

        拷贝构造函数也是如此,

        而operator=函数语法上可以是虚函数(不会报错),但无实际价值。因为将函数定义成虚函数是为了构成多态,而基类的派生类operator=函数的返回值始终不同,所以始终构成不了多态,所以无实际作用。

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

        若定义成了虚函数的同时也构成了多态,那么普通函数快,若没构成多态,则一样快。

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

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

后记

        通过以上的学习,想必大家也是知道我所说的易混淆之处在哪了,比如虚表与虚基表、重写与隐藏等等,这些都是要重点关注的知识点,也是面试当中经常会问到的点,除此之外,还有多态的原理,比较难以理解,需要多花些时间去理解。除了这些基本的知识点之外,最最重要的最后的常见问题,都是面试当中面试官的“杀手锏”,不专门去了解的话,还是很难回答的,建议在面试之前看一遍。以上就是多态相关的重点内容介绍了,好好学,拜拜!


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

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

相关文章

如何使用jenkins、ant、selenium、testng搭建自动化测试框架

如果在你的理解中自动化测试就是在eclipse里面讲webdriver的包引入&#xff0c;然后写一些测试脚本&#xff0c;这就是你所说的自动化测试&#xff0c;其实这个还不能算是真正的自动化测试&#xff0c;你见过每次需要运行的时候还需要打开eclipse然后去选择运行文件吗&#xff…

Linux Ubuntu命令行快速配置C++开发环境

本文介绍在Linux操作系统的Ubuntu版本中&#xff0c;基于命令行&#xff0c;快速配置C 编辑、编译、运行的代码开发环境的简便方法。 在之前的文章Linux操作系统Ubuntu 22.04配置Visual Studio Code与C代码开发环境的方法(https://blog.csdn.net/zhebushibiaoshifu/article/det…

C++标准模板库——vector的使用及其模拟实现

目录 一. vector的介绍 1.vector的介绍 二.vector的使用 vector中常见接口的介绍vector的构造和析构函数vector的三种遍历方式 三.vector的模拟实现 vector的增删查改vector容器的容量变化和大小增减vector迭代器失效问题vector的小框架 构造函数和析构函数迭代器和operat…

基于Java的高校竞赛管理系统设计与实现(亮点:发起比赛、报名、审核、评委打分、获奖排名,可随意更换主题如蓝桥杯、ACM、王者荣耀、吃鸡等竞赛)

高校竞赛管理系统 一、前言二、我的优势2.1 自己的网站2.2 自己的小程序&#xff08;小蔡coding&#xff09;2.3 有保障的售后2.4 福利 三、开发环境与技术3.1 MySQL数据库3.2 Vue前端技术3.3 Spring Boot框架3.4 微信小程序 四、功能设计4.1 主要功能描述4.2 系统角色 五、系统…

vue获取本地缓存并转为json格式

场景 要求获取当前登录用户id&#xff0c;传入后台去筛选属于该用户的数据&#xff1b; 当前登录用户信息一般会在本地存储中&#xff0c;有些则是在session中&#xff0c;此处只对本地存储做讨论&#xff1b; 本地缓存的用法 1 存储数据 localStorage.setltem(userId,"…

VS Code用AI写代码:Codeium插件

文章目录 Codeiumchat代码生成 Codeium Codeium是基于边缘计算的代码AI工具&#xff0c;提供超过70种编程语言的代码补全、对话、搜索等功能&#xff0c;相当霸道。 在插件栏搜索到Codeium之后&#xff0c;需要科学上网安装&#xff0c;安装完成后会提示注册。注册之后&#…

2023-9-22 滑雪

题目链接&#xff1a;滑雪 #include <cstring> #include <algorithm> #include <iostream>using namespace std;const int N 310;int n, m; int h[N][N]; int f[N][N];int dx[4] {-1, 0, 1, 0}, dy[4] {0, 1, 0, -1};int dp(int x, int y) {int &v f…

MySQL数据库简介+库表管理操作+数据库用户管理

Mysql Part 1 一、数据库的基本概念1.1 使用数据库的必要性1.2 数据库基本概念1.2.1 数据&#xff08;Data&#xff09;1.2.2 表1.2.3 数据库1.2.4 数据库管理系统&#xff08;DBMS&#xff09;1.2.5 数据库系统 1.3 数据库的分类1.3.1 关系数据库 SQL1.3.2 非关系数据库 NoSQL…

看阿里测试工程师如何玩转postman+newman+jenkins接口自动化

【软件测试面试突击班】如何逼自己一周刷完软件测试八股文教程&#xff0c;刷完面试就稳了&#xff0c;你也可以当高薪软件测试工程师&#xff08;自动化测试&#xff09; postman用来做接口测试非常方便&#xff0c;接口较多时&#xff0c;则可以实现接口自动化 一、环境准备…

【DLL修复工具下载】一键修复电脑丢失d3dcompiler_47.dll问题方法

在我们使用电脑的过程中&#xff0c;有时候会遇到一些错误提示&#xff0c;其中“缺失 d3dcompiler_47.dll”就是比较常见的一种。那么&#xff0c;d3dcompiler_47.dll 到底是什么呢&#xff1f;为什么会出现缺失的情况&#xff1f;丢失 d3dcompiler_47.dll 又会对电脑产生什么…

芋道商城,基于 Vue + Uniapp 实现,支持分销、拼团、砍价、秒杀、优惠券、积分、会员等级、小程序直播、页面 DIY 等功能

商城简介 芋道商城&#xff0c;基于 芋道开发平台 构建&#xff0c;以开发者为中心&#xff0c;打造中国第一流的 Java 开源商城系统&#xff0c;全部开源&#xff0c;个人与企业可 100% 免费使用。 有任何问题&#xff0c;或者想要的功能&#xff0c;可以在 Issues 中提给艿艿…

低代码助力企业数字化转型

在当今这个数字化快速发展的时代&#xff0c;企业面临的竞争越来越激烈&#xff0c;数字化转型已成为企业发展的必经之路。低代码平台作为一种新型的开发工具&#xff0c;正在逐渐成为企业数字化转型的重要助力。本文将从数字化转型背景、低代码平台介绍、低代码平台的应用、低…

SIEM 中的事件关联

什么是 SIEM 中的事件关联 SIEM 中的事件关联可帮助安全团队识别来自不同来源的安全事件并确定其优先级&#xff0c;从而提供更全面的整体安全环境视图。 在典型的 IT 环境中&#xff0c;会跨各种系统和应用程序生成大量事件和日志。孤立地看&#xff0c;其中许多事件可能看起…

聊一聊 TLS/SSL

哈喽大家好&#xff0c;我是咸鱼 当我们在上网冲浪的时候&#xff0c;会在浏览器界面顶部看到一个小锁标志&#xff0c;或者网址以 “https://” 开头 这意味着我们正在使用 TLS/SSL 协议进行安全通信。虽然它可能看起来只是一个小小的锁图标和一个 “https” &#xff0c;但…

记录一次错误---想让U-net网络输入大小不一致的图片

最近在看Deeplab系列的论文&#xff0c;文中提到了语义分割领域的一个难题是&#xff1a;将图片输入网络之前需要resize成统一大小&#xff0c;但是resize的话会造成细节信息的损失&#xff0c;所以想要网络处理任意大小的图片输入。我之前训练的U-net网络都是resize成224*224大…

Linux 本地 Docker Registry本地镜像仓库远程连接【内网穿透】

Linux 本地 Docker Registry本地镜像仓库远程连接 文章目录 Linux 本地 Docker Registry本地镜像仓库远程连接1. 部署Docker Registry2. 本地测试推送镜像3. Linux 安装cpolar4. 配置Docker Registry公网访问地址5. 公网远程推送Docker Registry6. 固定Docker Registry公网地址…

【CSS】画个三角形或圆形或环

首先通过调整边框&#xff0c;我们可以发现一些端倪 <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title></head><style>.box{width: 150px;height:150px;border: 50px solid black;}</style&g…

【docker安装Mysql并配置主从复制】

Mysql主从复制 目的&#xff1a; 是为了后面naocs集群的服务配置做准备工作 准备工作 准备至少两台虚拟机或服务器&#xff0c;安装好了docker&#xff0c;找到他们的ip地址 后面操作都用xshell操作来代替 拉取并启动mysql镜像和容器 主机的命令为mysql01&#xff0c;对…

机器学习笔记:seq2seq attentioned seq2seq

1 Seq2Seq 1.1 介绍 对于序列对<X,Y>&#xff0c;我们的目标是给定输入序列X&#xff0c;期待通过Encoder-Decoder框架来生成目标序列Y Encoder对输入的序列X进行编码&#xff0c;将输入序列通过非线性变换转化为中间语义表示C&#xff1a; Decoder根据序列X的中间语义…

百度知道本地搭建环境无限制采集聚合【最新版】

本工具是本地php环境搭建&#xff0c;根据关键词进行采集聚合某度知道&#xff0c;不限制ip&#xff0c;最新版新添加了违规词过滤&#xff0c;样式处理&#xff0c;自动匹配优质标题等功能&#xff0c;只需要导入关键词可以无限采集&#xff0c;是养站的好帮手&#xff01; 功…