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

三大特性之多态

文章目录

    • 静态的多态
    • 动态的多态
      • 虚函数
        • 虚函数的重写(覆盖)
        • 利用虚函数重写实现多态
        • 重写的两个例外
          • 1.协变
          • 2.析构函数的函数名不同
        • C++11的override和final
      • 重载,重写(覆盖),重定义(隐藏)
      • 抽象类
      • 接口继承和实现继承
      • 多态实现的原理
        • 虚函数表
        • 虚函数表的存储位置
      • 一些关于多态的问题

多态是不同继承关系的类对象去调用同一个函数,产生了不同效果的行为。

静态的多态

调用同一个函数,产生不同效果的行为,这不就是函数重载吗!函数重载其实是一种静态的多态,相同的函数名传不同的参数调用的函数也就不同,但是调用哪个函数是在编译阶段就已经被确定好了。函数重载是一种编译时绑定,也就是静态绑定。常用的流插入和流体取也是一种函数重载

动态的多态

动态的多态才是本篇文章中要讲的主要内容,它在调用函数时与与类型无关而是与它所存放的对象有关(普通调用是按类型)。具体调用哪个函数运行时才知道,又叫运行时绑定,也就是动态绑定。

多态的构成必须要满足两个条件:

1.必须要通过父类的引用或者指针作为形参来调用

为什么一定要是父类的引用或者指针,对于这个问题《深度探索C++模型》中这样说:“一个pointer或一个reference之所以支持多态,是因为它们并不引发内存任何“与类型有关的内存委托操作; 会受到改变的。只有它们所指向内存的大小和解释方式 而已”。

对于上述话可以这样理解:

  1. 指针和引用类型只是要求了基地址和这种指针所指对象的内存大小,与对象的类型无关,相当于把指向的内存解释成指针或引用的类型。
  2. 如果直接把子类对象赋值给父类对象,就牵扯到内存模型,编译器就会回避虚机制,无法达到多态的效果。

2.被调用的函数必须是虚函数

虚函数

所谓的虚函数就是被virtual关键字所修饰的函数

class Person 
{
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }//这就是虚函数
};

虽然虚函数和虚继承都使用了virtual关键字,但是两者之间没有任何联系

虚函数的重写(覆盖)

如果子类中有和父类一样的虚函数(返回值类型,函数名称,参数列表相同),那么就称该子类的虚函数重写了父类的虚函数。

这里有一点需要注意:如果父类在声明的时候加了virtual,即使子类在声明同名函数时不加virtual也会完成重写(可以理解为子类在继承父类时将虚属性也继承下来了),但这样写是不规范的,建议不要这样写。

虚函数的重写也可以被称为虚函数的覆盖,因为带有虚函数的类都有一个虚函数表,在继承的时候子类会继承父类的虚函数表,如果子类对某一个虚函数进行重写了,那么该虚函数在子类的虚函数表中就会被重写的虚函数覆盖。

利用虚函数重写实现多态

class Person
{
public:virtual void BuyTicket() { cout << "买票-全价" << endl; }
};
class Student : public Person 
{
public:virtual void BuyTicket() { cout << "买票-半价" << endl; }};
void Func(Person& p)
{p.BuyTicket();
}
int main()
{Person ps;Student st;Func(ps);Func(st);return 0;
}

在这里插入图片描述

可以看到虽然我都是调用Func函数,但最后Func函数帮我调用到了不同的函数,这就是多态。

有了多态以后在调用函数的时候首先要看该函数是否构成多态,如果构成多态那么就不用考虑类型,只需要看该变量中存放的是何种对象,按照对象去调用函数;如果不构成多态,那么就只看类型,无论该变量中存的是何对象都不影响,只看类型调用(这里如果不是多态就都调用Person的函数)。

重写的两个例外

1.协变

子类对于父类函数的重写,返回类型可以不同,但必须要是返回父子类关系的指针或引用(即基类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用),称为协变

class Person 
{
public:virtual A* f() {return new A;}
};
class Student : public Person 
{
public:virtual B* f() {return new B;}
};
2.析构函数的函数名不同

析构函数的函数名要求必须要与类名相同,也就是说各个类的析构函数名都不同,但是其实编译器会将析构函数的函数名统一处理成destructor。

析构函数不但能重写,并且析构函数建议定义成虚函数

如果我定义了一个子类的对象,并将该子类对象赋值给一个父类的指针,当我释放父类的时候只会调用父类的析构函数,也就是说只释放了子类中父类的那一部分资源,而没有释放子类的资源,这就可能会导致内存泄漏。

如果我将析构函数定义为虚函数并重写,那么我在释放父类指针的时候,调用的是子类的析构函数,子类析构函数对于父类那一部分资源通过父类的析构函数清理,同时也会清理自己的资源。

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

在这里插入图片描述

C++11的override和final

1.被final关键字修饰的函数不能被重写

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

在这里插入图片描述

2.使用override关键字检查该函数是否被重写,如果没有重写就报错

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

重载,重写(覆盖),重定义(隐藏)

重载

1.要在同一个作用域中

2.函数名相同,参数列表相同,返回值可以不同

重写(覆盖)

1.两个函数分别在父类和子类的作用域中

2.返回值相同(协变除外),函数名相同,参数列表相同

3.只有虚函数才构成重写

重定义(隐藏)

1.两个函数分别在父类和子类的作用域中

2.函数名相同只要不构成重写就是重定义

抽象类

与虚函数对应的还有一个纯虚函数,只要在虚函数声明的最后加上=0那么这个虚函数就变成了纯虚函数。如果一个类包含纯虚函数,那么这个类就是抽象类。抽象类不能实例化对象,并且如果继承抽象类的子类不对纯虚函数进行重写的话,子类也是一个抽象类无法实例化对象。纯虚函数规范了子类必须重写,此外纯虚函数更体现了接口继承。

class A
{
public:virtual void test() = 0;int _a;
};class B :public A
{
public:virtual void test(){cout << "重写了纯虚函数" << endl;}int _b;
};
int main()
{A a;B b;return 0;
}

在这里插入图片描述

接口继承和实现继承

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

多态实现的原理

首先我们来计算一下下面这个类的大小

class A
{
public:virtual void test();
}

按照我们类和对象阶段所说,一个没有成员变量的类就是空类,空类的大小为1字节,用来占位。

但这个类的大小为4字节

在这里插入图片描述

这是因为如果一个类中有虚函数,那么该类中会有一个隐藏的指针,该指针指向一个虚函数表。

虚函数表

在这里插入图片描述

可以看到虚函数表中存放的是虚函数的地址,所谓虚函数的重写其实就是将重写过的虚函数的地址覆盖到原虚函数地址上。

在这里插入图片描述

因为每个类都有自己独立的虚函数表,所以不同的类对象就可以通过不同的虚函数表访问到不同的虚函数。

在vs下虚函数表都是以空结尾,但是Linux下就不是;

一个变量中如果存放的是子类的对象,那么该变量中的前四个字节就是子类所对应的虚函数表,该表中存放的是子类所对应的虚函数。如果放的是父类的对象,那么该变量的前四个字节就是父类的虚函数表,其中存放的也就是父类所对应虚函数的地址。多态的产生也就是根据对象的前四个字节找到虚函数表,调用其中的虚函数实现的。

虚函数表的存储位置

此外虚函数表存储在常量区中(代码段),通过下面的代码可以看到各个区域的地址,虚表的地址和哪个最为接近就可以说明虚表在哪个区域,经观察确实是在代码段上。

class Base
{
public:virtual void Func1(){cout << "Base::Func1()" << endl;}virtual void Func2(){cout << "Base::Func2()" << endl;}void Func3(){cout << "Base::Func3()" << endl;}private:int _b = 1;char _ch;
};class Derive : public Base
{
public:virtual void Func1(){cout << "Derive::Func1()" << endl;}void Func3(){cout << "Derive::Func3()" << endl;}
private:int _d = 2;
};
int main()
{int a = 0;cout << "栈:" << &a << endl;int* p1 = new int;cout << "堆:" << p1 << endl;const char* str = "hello world";cout << "代码段/常量区:" << (void*)str << endl;static int b = 0;cout << "静态区/数据段:" << &b << endl;Base be;cout << "虚表:" << (void*)*((int*)&be) << endl;Base* ptr1 = &be;int* ptr2 = (int*)ptr1;printf("虚表:%p\n", *ptr2);Derive de;cout << "虚表:" << (void*)*((int*)&de) << endl;Base b1;Base b2;return 0;
}

补充

子类不但有从父类继承下来的成员,还有子类自己特有的成员,也就是说子类除了有重写父类的虚函数还有自己特有的虚函数。

对于单继承的类来说:子类特有的虚函数和重写的虚函数放在同一张虚函数表中

对于多继承的子类来说:它继承了多少个有虚函数的类就有几张虚函数表,但子类所特有的虚函数会放在第一张虚函数表中

一些关于多态的问题

  1. 什么是多态

答:多态就是多种形态,不同的对象来做同一件事产生不同效果的行为就可以称为多态。

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

答:可以,不过编译器就忽略inline属性,这个函数就不再是inline,因为虚函数要放到虚表中去。

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

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

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

答:不能,因为对象中的虚函数表指针是在构造函数初始化列表阶段才初始化的。

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

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

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

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

相关文章:

三大特性之多态

文章目录 静态的多态动态的多态虚函数虚函数的重写&#xff08;覆盖&#xff09;利用虚函数重写实现多态重写的两个例外1.协变2.析构函数的函数名不同 C11的override和final 重载&#xff0c;重写&#xff08;覆盖&#xff09;&#xff0c;重定义&#xff08;隐藏&#xff09;抽…...

单调队列优化dp

文章目录 单调队列优化dp烽火传递修剪草坪绿色通道琪露诺旅行问题Watching Fireworks is Fun瑰丽华尔兹股票交易 单调队列优化dp 文章首发于我的个人博客&#xff1a;欢迎大佬们来逛逛 单调队列优化dp的建模形式&#xff1a;这是窗口右滑动的情况 对于窗口左滑动的也是同理。…...

【低压配电漏电继电器660V/LLJ-100H/AC220V 中性点漏电保护 JOSEF】

LLJ-F(S)系列漏电继电器 系列型号&#xff1a; LLJ-10F(S)漏电继电器LLJ-15F(S)漏电继电器LLJ-16F(S)漏电继电器 LLJ-25F(S)漏电继电器LLJ-30F(S)漏电继电器LLJ-32F(S)漏电继电器 LLJ-60F(S)漏电继电器LLJ-63F(S)漏电继电器LLJ-80F(S)漏电继电器 LLJ-100F(S)漏电继电器LLJ-120…...

[数据结构习题]栈——中心对称链

[数据结构习题]栈——中心对称链 &#x1f449;知识点导航&#x1f48e;&#xff1a;【数据结构】栈和队列 &#x1f449;[王道数据结构]习题导航&#x1f48e;&#xff1a; p a g e 70.4 page70.4 page70.4 本节为栈和链表综合练习题 题目描述&#xff1a; &#x1f387;思路…...

AMD Software Adrenalin Edition 23.5.1驱动发布,快速获取驱动

AMD新驱动赶在五月天发布&#xff01;AMD Software Adrenalin Edition 23.5.1驱动 &#xff0c;为部分游戏带来支持&#xff0c;以及为重要的软件带来修复。驱动人生带大家一览AMD WHQL 23.5.1驱动的优化内容。 游戏方面&#xff0c;AMD WHQL 23.5.1主要为游戏《指环王&#x…...

Visual Studio内引用Lua解释器,编译Lua源码,执行Lua脚本

前言 本篇在讲什么 在Visual Studio中引入lua的解释器 使用C调用Lua文件 本篇适合什么 适合初学Lua的小白 适合需要C/C和lua结合开发的人 本篇需要什么 对Lua语法有简单认知 对C/C语法有简单认知 依赖Lua5.1的环境 依赖VS 2017编辑器 本篇的特色 具有全流程的图文…...

【赏】C语言迷宫游戏设计如何解决屏幕严重刷屏问题同时实现运行时间的显示

要解决屏幕严重刷屏问题,可以参考以下方法: 在每次刷新前清空屏幕,使用system("cls")命令来实现清屏。 只在需要更新的地方进行刷新,而不是整个屏幕都重新绘制。在此代码中,只需要在用户输入移动指令后更新电子鼠的位置即可,不用每次循环都重新画整个迷宫。同时…...

Spring Boot如何实现接口文档自动生成

Spring Boot如何实现接口文档自动生成 在开发Web应用程序时&#xff0c;接口文档是非常重要的一环&#xff0c;它可以帮助我们快速了解API的功能和使用方法&#xff0c;同时也是与其他开发人员和团队协作的重要工具。然而&#xff0c;手动编写和维护接口文档是一项繁琐的工作&…...

二进制概述-0day漏洞利用原理(1)

二进制利用基本原理,Lord PE的使用,凡是资源性的物质且可表达的皆可利用。 往期文章: 漏洞概述-0day漏洞利用原理(0)_luozhonghua2000的博客-CSDN博客 PE 文件格式 PE (Portable Exec utable) 是 Win32 平台下可执行文件遵守的数据格式。常见的可执行文件(如“*.exe”文件…...

加密与解密 调试篇 动态调试技术 (二)-常见断点

目录 常见的断点 1.INT 3 断点 检测 绕过 2.硬件断点 原理 我们给出硬件中断的例子 删除硬件断点 3.内存断点 原理 例子 删除 区别 总结 4.内存访问一次性断点 5.消息断点 例子 删除 6.条件断点 &#xff08;1&#xff09;按寄存器条件中断 &#xff08;2&…...

【JavaScript】拾遗(5.25)

文章目录 1. JavaScript2.HTML嵌入JS的第一种方式:行间事件3.HTML嵌入JS的第二种方式:脚本块的方式4. HTML嵌入JS的第三种方式:外部式(外链式)5. 局部变量和全局变量6. 函数7.事件8.回调函数8.1 注册事件8.2 代码的执行顺序 1. JavaScript JavaScript是一门脚本语言。&#xf…...

QMI8658 - 姿态传感器学习笔记 - Ⅲ

文章目录 1.复位1.1 上电复位&#xff1a;1.2 推荐工作条件 2. 校准(COD)2.1 校准步骤2.2 校准注意事项&#xff1a;2.3 校准状态指示2.4 校准参数更新 3. 自检3.1 加速度计自检3.2 陀螺仪自检 4. Ctrl94.1 写Ctrl94.2 读Ctrl94.3 Ctrl9详细命令说明 5. 中断5.1 同步采样模式5.…...

PHP+vue二手车交易信息网站系统

原来二手车网站由于二手车网站制度的不完善&#xff0c;许多城市的二手车网站市场都很少&#xff0c;而且欺诈行文较严重&#xff0c;肆意提高价格&#xff0c;隐瞒汽车所存在的故障问题&#xff0c;人们买卖二手车还是经过朋友帮忙介绍的途径来实现。这就导致了很多人的想卖车…...

NTM中attr的用法

代码1 attrs class CopyTaskParams(object):name attrib(default"copy-task")controller_size attrib(default100, convertint)controller_layers attrib(default1,convertint)num_heads attrib(default1, convertint)sequence_width attrib(default8, convert…...

【python资料】pandas的条件查询

一、说明 在使用Pandas的DataFrame进行数据挖掘的时候&#xff0c;需要形形色色的条件查询&#xff0c;但是这些查询的基本语法是啥&#xff0c;查询的灵活性如何&#xff0c;本文将对他们进行详细列出&#xff0c;便于以后查阅。 二、Pandas条件查询方法 2.1 简单条件查询 1、…...

中间件(三)- Kafka(二)

Kafka 6. 高效读写&Zookeeper作用6.1 Kafka的高效读写6.2 Kafka中zookeeper的作用 7. 事务7.1 Producer事务7.2 Consumer事务 8. API生产者流程9. 通过python调用kafka9.1 安装插件9.2 生产者&#xff08;Producer&#xff09;与消费者&#xff08;Consumer&#xff09;9.3…...

DAY01_MySQL基础数据类型navicat使用DDL\DML\DQL语句练习

目录 1 数据库相关概念1.1 数据库1.2 数据库管理系统1.3 常见的数据库管理系统1.4 SQL 2 MySQL2.1 MySQL安装2.1.1 安装步骤 2.2 MySQL配置2.2.1 添加环境变量2.2.2 MySQL登录2.2.3 退出MySQL 2.3 MySQL数据模型2.4 MySQL目录结构2.5 MySQL一些命令2.5.1 修改默认账户密码2.5.2…...

数据安全复合治理框架和模型解读(0)

数据治理,数据安全治理行业在发展,在实践,所以很多东西是实践出来的,哪有什么神仙理论指导,即使有也是一家之说,但为了提高企业投产比,必要的认知是必须的,当前和未来更需要专业和创新。数据安全治理要充分考虑现实数据场景,强化业务安全与数据安全治理,统一来治理,…...

Java程序设计入门教程--逻辑运算符和位运算符

目录 逻辑运算符 位运算符 逻辑运算符 逻辑运算符就是表示逻辑关系的运算符。下表列出了逻辑运算符的基本运算&#xff0c;假设布尔变量A为真&#xff0c;变量B为假。 逻辑运算符表 操作符 描述 例子 && 当且仅当两个操作数都为真&#xff0c;条件才为真。 &…...

接口测试简介以及接口测试用例设计思路

接口测试简介 1.什么是接口 接口就是内部模块对模块&#xff0c;外部系统对其他服务提供的一种可调用或者连接的能力的标准&#xff0c;就好比usb接口&#xff0c;他是系统向外接提供的一种用于物理数据传输的一个接口&#xff0c;当然仅仅是一个接口是不能进行传输的&#x…...

KubeSphere 容器平台高可用:环境搭建与可视化操作指南

Linux_k8s篇 欢迎来到Linux的世界&#xff0c;看笔记好好学多敲多打&#xff0c;每个人都是大神&#xff01; 题目&#xff1a;KubeSphere 容器平台高可用&#xff1a;环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...

深入剖析AI大模型:大模型时代的 Prompt 工程全解析

今天聊的内容&#xff0c;我认为是AI开发里面非常重要的内容。它在AI开发里无处不在&#xff0c;当你对 AI 助手说 "用李白的风格写一首关于人工智能的诗"&#xff0c;或者让翻译模型 "将这段合同翻译成商务日语" 时&#xff0c;输入的这句话就是 Prompt。…...

从零实现STL哈希容器:unordered_map/unordered_set封装详解

本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说&#xff0c;直接开始吧&#xff01; 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...

在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用

1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...

用docker来安装部署freeswitch记录

今天刚才测试一个callcenter的项目&#xff0c;所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...

Device Mapper 机制

Device Mapper 机制详解 Device Mapper&#xff08;简称 DM&#xff09;是 Linux 内核中的一套通用块设备映射框架&#xff0c;为 LVM、加密磁盘、RAID 等提供底层支持。本文将详细介绍 Device Mapper 的原理、实现、内核配置、常用工具、操作测试流程&#xff0c;并配以详细的…...

深度学习水论文:mamba+图像增强

&#x1f9c0;当前视觉领域对高效长序列建模需求激增&#xff0c;对Mamba图像增强这方向的研究自然也逐渐火热。原因在于其高效长程建模&#xff0c;以及动态计算优势&#xff0c;在图像质量提升和细节恢复方面有难以替代的作用。 &#x1f9c0;因此短时间内&#xff0c;就有不…...

【从零学习JVM|第三篇】类的生命周期(高频面试题)

前言&#xff1a; 在Java编程中&#xff0c;类的生命周期是指类从被加载到内存中开始&#xff0c;到被卸载出内存为止的整个过程。了解类的生命周期对于理解Java程序的运行机制以及性能优化非常重要。本文会深入探寻类的生命周期&#xff0c;让读者对此有深刻印象。 目录 ​…...

GitFlow 工作模式(详解)

今天再学项目的过程中遇到使用gitflow模式管理代码&#xff0c;因此进行学习并且发布关于gitflow的一些思考 Git与GitFlow模式 我们在写代码的时候通常会进行网上保存&#xff0c;无论是github还是gittee&#xff0c;都是一种基于git去保存代码的形式&#xff0c;这样保存代码…...

BLEU评分:机器翻译质量评估的黄金标准

BLEU评分&#xff1a;机器翻译质量评估的黄金标准 1. 引言 在自然语言处理(NLP)领域&#xff0c;衡量一个机器翻译模型的性能至关重要。BLEU (Bilingual Evaluation Understudy) 作为一种自动化评估指标&#xff0c;自2002年由IBM的Kishore Papineni等人提出以来&#xff0c;…...