C++多态特性详解

news2025/1/6 19:12:29

目录

概念:

定义及实现:

虚函数重写的两个例外:

1.协变:

 2.析构函数的重写:

final关键字:

 override关键字:

多态是如何实现的(底层):

面试题:


概念:

        所谓多态就是,当去完成某个行为时,不同的对象会产生不同的状态,导致不同的结果。

举个简单的例子:当去一个景区旅游,成人买票是全价,儿童买票是半价,这就是一种典型的多态。还有个典型的例子:大家在使用某多多助力拿红包时,有的能拿到,有的却邀了几百个人还是拿不到,这可能就是一种多态,如果你是新用户,它就会让你拿到,如果你是老用户,它可能让你拿不到。(仅个人猜测)

定义及实现:

c++中构成多态需要满足2个条件:

1.父类的指针或者引用去调用虚函数。

2.完成虚函数的重写,满足三同(函数名,返回值,参数)。

上述的三同有特例,后面会讲,我们先来简单实现一下上面买票这个例子的多态:

 试试传子类的引用看看结果如何:

注意:虚函数不能写成全局的,只能写在类里面。虚函数和正常的成员函数一样都存在代码段。

虚函数重写的两个例外:

        上面讲到了虚函数重写需要满足三同,函数名,返回值,形参都要保证和父类里面的对应虚函数相同,但也有两个例外。

1.协变:


        子类重写父类虚函数时,与父类虚函数返回值类型不同。即父类虚函数返回父类对象的指针或者引用,子类虚函数返回子类对象的指针或者引用时,称为协变。(基类与派生类虚函数返回值类型不同)

 

不一定返回自己的父类或者子类,别的父类或者子类也可以:

 2.析构函数的重写:

        如果父类的析构函数为虚函数,此时子类析构函数只要定义,无论是否加virtual关键字,
都与父类的析构函数构成重写,虽然基类与派生类析构函数名字不同。虽然函数名不相同,
看起来违背了重写的规则,其实不然,这里可以理解为编译器对析构函数的名称做了特殊处
理,编译后析构函数的名称统一处理成destructor。

看下面这个场景:

因为有切片的概念,父类的指针可以指向子类,注意看这里调用了两次父类的析构,但我们new出来的时一个父类和一个子类,这里看似没什么影响,但我们稍微对子类修改一下 :

这时如果不调用子类的析构就会造成内存泄漏,解决办法:

 结论:建议将析构函数写成虚函数,防止内存泄漏。

final关键字:

        当final修饰虚函数时,则改虚函数不能被重写:

        当final修饰一个类时,这个类为最终类,无法被继承:

 override关键字:

        加到子类重写的虚函数检查是否完成重写:

多态是如何实现的(底层):

 看下面这个场景

A里面明明只有一个变量_a,虚函数func是存在代码段的,没在类里面,按理来说应该算出4字节为何这里是16字节?

答:一旦类里面有虚函数,类的头4个(32位机器)或者8个(64位机器)字节就会有一个虚表指针 ,我这里是64位机器,然后又要满足结构体对齐,所以是16,我们用监视窗口看一下:

确实有个虚表指针,它指向该类虚函数的地址:

 想必已经猜出来了,虚表指针,指向的就是该类中所有虚函数的起始地址。

相同类型的类共用一个虚表,也就是虚表指针是相同的。 

当我们在子类对虚函数进行重写后,子类的虚表就存的是重写后的虚函数,父类的虚表存的还是重写前的虚函数,当我们用父类或者子类的引用或者指针去访问虚表时,就会访问到不同的虚表,从而完成不同的行为,构成多态。

那要是多继承,虚表指针是怎么存的呢?

再复杂点,多继承中的菱形继承又是怎么存虚表指针的呢?

此时D中的结构就如下图:

 再再再复杂一点,多继承中的菱形虚拟继承又是怎么存虚表指针的?

 此时D中的结构就如下图:

虚基表指针的知识我以后会讲 。

注意:虚表是存在常量区的,不是存在类里面,只是虚表指针存在类的头4个或者8个字节。 

面试题:

该程序运行的结果是什么?

A: A->0 B: B->1 C: A->1 D: B->0 E: 编译出错 F: 以上都不正确
答案:B

因为虚函数的重写,重写的只是定义,它的形参的缺省值用的还是父类里面的。 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码审计之浅谈RASP技术

前言: 想摆会烂,所以就落个笔吧。 其实本来是想写关于iast技术的,但是认真思考了下,感觉笔者自己本身也不太能讲清楚iast技术,怕误人子弟。 所以最后还是基于笔者的理解以及实际应用写一篇关于RASP技术的文章&#xf…

酷我音乐无广VIP破姐版,支持蝰蛇音效,PC端 v9.2.0.0

01 软件介绍 酷我音乐是一款优质在线音乐播放服务软件,拥有全面的音乐版权库存,中国好声音、蒙面歌王、燃烧吧少年等热门综艺的版权,包括无损音质选项,并涵盖广泛多样的音乐类型,如华语、英语、粤语音乐,以…

[暂未实现]APP签名不同保留数据覆盖安装记录

APP签名不同无法直接覆盖安装 使用adb可以卸载应用同时保留数据,但签名不同也无法覆盖安装(安装原来签名的应用打开和卸载前一样) 使用adb导出应用数据(QQ)db文件只有1kb,显然此方法也行不通

Superset二次开发之Legend功能优化

背景 Legend数据太长,影响整体图表体验,为改善用户体验,需要实现:1.数据省略展示,‘...’表示,鼠标悬停时,展示完整信息 2:文本内容从左向右滚动展示 柱状图优化 柱状图来自第三方Echarts插件,效果展示 功能核心在于红框的内容 option = {tooltip: {trigger: item,ax…

== 和 equals()区别,equals()重写问题

对于引用类型:比较的是两个引用是否相同(所指的是否为同一个对象),注:如果两个引用所指的对象内容一样,但是不是同一个对象(hashcode不一样),依然返回false,随…

Linux的基本指令(下)

各位大佬好 ,这里是阿川的博客 , 祝您变得更强 个人主页:在线OJ的阿川 大佬的支持和鼓励,将是我成长路上最大的动力 阿川水平有限,如有错误,欢迎大佬指正 这篇博客续博主的上篇博客Linux基本指令。 07 …

美团KV存储squirrel和Celler学习

文章目录 美团在KV存储squirrel优化和改进在水平方向1、对Gossip协议进行优化 在垂直扩展方面1、forkless RDB数据复制优化2、使用多线程,充分利用机器的多核能力 在高可用方面 美团持久化kv存储celler优化和改进水平扩展优化1、使用bulkload进行数据导入2、线程模型…

安卓手机运行 Windows 操作系统:一

在折腾上一篇文章的时候,发现了一条有趣的折腾分支,在这台老设备上运行 Windows 操作系统。 看起来应该蛮有趣的,那么就折腾一下吧。 写在前面 最早知道 WoA(Windows on ARM) 项目,是 2020 看到这篇报道…

Mysql如何通过ibd文件恢复数据

Mysql ibd文件恢复注意问题 创建数据库(随意创建)创建数据表(备注:表结构要和要恢复的表结构一致,row_format要和ibd文件的row_format一致,否则,会提示两者不一致。 当前row_formatdynamic&…

Day30:热帖排行、生成长图、将文件上传到云服务器、优化热门帖子列表、压力测试

热帖排行 不同的算分方式: 只存变化的帖子到redis中,每五分钟算一次分,定时任务 存redis 构建redis键 //统计帖子分数 //key:post:score -> value:postId public static String getPostScoreKey() {return PREFIX_POST SPLIT "…

vue2人力资源项目4路由和部门新增

组织架构路由 import layout from /layout export default {path: /department,component: layout, // 一级路由children: [{path: , // 二级路由地址为空 表示/department 显示一级路由二级路由component: () > import(/views/department),name: , // 可以用来跳转&#xf…

C#技巧之同步与异步

区别 首先,同步就是程序从上往下顺序执行,要执行完当前流程,才能往下个流程去。 而异步,则是启动当前流程以后,不需要等待流程完成,立刻就去执行下一个流程。 同步示例 创建一个窗体,往窗体里…

ES数据存储与查询基本原理

Elasticsearch(ES)简介 Elasticsearch(ES)是一个分布式、可扩展、近实时的搜索和分析引擎,它基于Lucene,设计用于云计算中,处理大规模文档检索和数据分析任务,常用于实现内部搜索引…

二维泊松方程(Neumann+Direchliet边界条件)有限元Matlab编程求解|程序源码+说明文本

专栏导读 作者简介:工学博士,高级工程师,专注于工业软件算法研究本文已收录于专栏:《有限元编程从入门到精通》本专栏旨在提供 1.以案例的形式讲解各类有限元问题的程序实现,并提供所有案例完整源码;2.单元…

详解面向对象-类和对象

1.面向对象与面向过程的区别 ①面向过程 :关注点是在实现功能的步骤上面,就是分析出解决问题所需要的步骤,让后函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以。对于简单的流程是适合面向过程的方式进行的&#x…

模型全参数训练和LoRA微调所需显存的分析

大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…

TypeScript 基础学习笔记:泛型 <T> vs 断言 as

TypeScript 基础学习笔记&#xff1a;泛型 <T> vs 断言 as &#x1f525; 引言 &#x1f44b; TypeScript (TS) 以其静态类型的魔力&#xff0c;让我们的代码更加健壮、易读且易于维护。今天&#xff0c;我们将深入探讨两个核心概念——泛型&#xff08;Generics&#x…

使用socket+Python实现ping

import os import socket import struct import select import time# 计算校验和&#xff0c;用于确保数据的完整性 def checksum(source_string):sum 0count 0max_count len(source_string)# 处理成对的字节while count < max_count - 1:val source_string[count 1] *…

结构体介绍(2)

结构体介绍&#xff08;2&#xff09; 前言一、结构体的内存对齐之深入理解为什么存在内存对齐&#xff1f;修改默认对齐数 二、结构体传参2.1&#xff1a;该怎么传参呢&#xff1f; 三、结构体实现位段3.1什么是位段位段的内存分配位段的跨平台问题 总结 前言 根据之前讲了结…

金融行业AI大模型百项应用案例综述【大模型系列】

逐浪金融大模型的玩家&#xff0c;除了BAT、华为等高科技巨头&#xff0c;试图以技术优势充当产业链的“卖铲人”&#xff0c;更多的还是金融和类金融企业&#xff0c;包括银行、保险、互金、券商等&#xff0c;既不想被喧宾夺主&#xff0c;又不想肥水外流&#xff0c;都在押注…