【C++进阶篇】多态

news2025/5/13 5:58:04

深入探索C++多态:静态与动态绑定的奥秘

  • 一. 多态
    • 1.1 定义
    • 1.2 多态定义及实现
      • 1.2.1 多态构成条件
        • 1.2.1.1 实现多态两个必要条件
        • 1.2.1.2 虚函数
        • 1.2.1.3 虚函数的重写/覆盖
        • 1.2.1.4 协变
        • 1.2.1.5 析构函数重写
        • 1.2.1.6 override和final关键字
        • 1.2.1.7 重载/重写/隐藏的对⽐
    • 1.3 纯虚函数和抽象类
    • 1.4 多态原理
      • 1.4.1 虚函数表指针
      • 1.4.2 原理
      • 1.4.3 静态绑定与动态绑定
      • 1.4.4 虚函数表
  • 二. 最后

本文章介绍面向对象编程的三大特性中的多态之一。

💬 欢迎讨论:如果你在学习过程中有任何问题或想法,欢迎在评论区留言,我们一起交流学习。你的支持是我继续创作的动力!
👍点赞、收藏与分享:觉得这篇文章对你有帮助吗?别忘了点赞、收藏并分享给更多的小伙伴哦!你们的支持是我不断进步的动力!
🚀分享给更多人:如果你觉得这篇文章对你有帮助,欢迎分享给更多对 C++ 感兴趣的朋友,让我们一起进步!

一. 多态

1.1 定义

多态使得不同类型的对象可以通过相同的接口进行交互,而每个对象的具体行为是根据其自身类型来决定的,分为编译时多态和运行时多态。举个例子:对于普通人来说,票价全价,学生来说,票价半价,军人来说,优先买票等,指在根据不同的对象完成不同买票行为。
下面用示例代码来展现该行为(如下):

#include <iostream>
using namespace std;

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

class Student : public Person {
public:
	virtual void BuyTicket() { cout << "买票-打折" << endl; }
};

void Func(Person* ptr)
{
	// 这里可以看到虽然都是Person指针Ptr在调用BuyTicket
	// 但是跟ptr没关系,而是由ptr指向的对象决定的。
	ptr->BuyTicket();
}

int main()
{
	Person p1;
	Student s1;
	Func(&p1);
	Func(&s1);
	return 0;
}

结果如下:
在这里插入图片描述
通过传递不同的对象,来实现不同的行为,上述传递Person类和Student类,分别对应的行为是全价和半价。

1.2 多态定义及实现

1.2.1 多态构成条件

多态的实现原理通过虚函数表,虚表由虚函数构成,即 Virtual 修饰的函数,还需要虚表指针进行定位函数调用。

1.2.1.1 实现多态两个必要条件
  • 必须是基类的指针或引用调用虚函数
  • 被调用的函数必须是虚函数,并且派生类需对基类被调用的虚函数重写/覆盖
1.2.1.2 虚函数

注意:必须是成员函数,不是成员函数的报错示例:
在这里插入图片描述
error: 类声明的外部说明符无效。

定义:类成员函数被virtual修饰,这个成员函数被称为虚函数。

1.2.1.3 虚函数的重写/覆盖

定义:派生类有一个与基类完成相同的虚函数(即派生类函数与基类函数的返回值类型、函数名字,参数列表完全形同,缺一不可),称为派生类的虚函数重写了基类的虚函数。注意:派生类的虚函数不加 virtual 修饰,也构成重写,但是不规范,不建议使用。

  • 例题:
class A
{
public:
virtual void func(int val = 1){ std::cout<<"A->"<< val <<std::endl;}
virtual void test(){ func();}
};
class B : public A
{
public:
void func(int val = 0){ std::cout<<"B->"<< val <<std::endl; }
};
int main(int argc ,char* argv[])
{
B*p = new B;
p->test();
return 0;
}

以下程序输出结果是什么()
A: A->0 B: B->1 C: A->1 D: B->0 E: 编译出错 F: 以上都不正确

  • 解释:

调用func前面是A*类型,是基类的指针调用,派生类对该函数进行了重写,所以构成多态。指向派生类,所以调用派生类的func,而派生类B前面省略了virtual,编译器会将派生类调用的func修改成:virtual void func(int val = 1){ std::cout<<“B->”<< val <std::endl;}所以叫作对基类虚函数的重写。即B-1,所以答案是B。

1.2.1.4 协变

派生类重写基类虚函数时,与基类虚函数返回值类型不同时。简单点说就是:基类指针返回基类对象的指针或引用,派生类返回派生类对象的指针或引用,称为协变。

  • 示例代码:
class A {};
class B : public A {};

class Person {
public:
	virtual A* BuyTicket()//返回基类指针对象
	{
		cout << "买票-全价" << endl;
		return nullptr;
	}
};

class Student : public Person {
public:
	virtual B* BuyTicket()//返回派生类指针对象
	{
		cout << "买票-打折" << endl;
		return nullptr;
	}
};

void Func(Person* ptr)
{
	ptr->BuyTicket();
}

int main()
{
	Person ps;
	Student st;

	Func(&ps);
	Func(&st);

	return 0;
}

上述举了一个关于协变的例子。

1.2.1.5 析构函数重写

注意:需将对基类的析构函数进行重写。下面给个代码例子来解释:

class A
{
public:
	 virtual ~A()
	{
		cout << "~A()" << endl;
	}
};

class B : public A {
public:
	virtual ~B() override 
	{
		cout << "~B()->delete:" << _p << endl;
		delete [] _p;
	}
protected:
	int* _p = new int[10];
};

int main()
{
	A* p1 = new A;
	A* p2 = new B; //这是C++的静态绑定特性:非虚函数调用仅由指针的静态类型(此处为A * )决定。

	delete p1;
	delete p2;

	return 0;
}

注意:C++的静态绑定特性:非虚函数调用仅由指针的静态类型决定。当基类的析构函数没有被修饰为虚函数,进行静态绑定,非虚函数调用仅由指针的静态类型(A*)决定。下面的B对象的资源不会被释放,造成内存泄漏。。当基类的析构函数被修饰为虚函数,构成多态,运行时绑定p2指针指向B对象,调用B对象的析构函数,成功将B对象的资源释放。

所以建议:最好将基类的析构函数修饰为虚函数,避免内存泄漏。

1.2.1.6 override和final关键字
  • override:检查成员函数是否构成重写
  • final:被修饰的虚函数不能被重写
1.2.1.7 重载/重写/隐藏的对⽐
  1. 重载:两个函数在同一作用域,函数名相同,参数不同,参数类型或者个数不同,返回值不关心,即可构成重载。
  2. 重写:两个函数作用域在不同的作用域(一般是基类和派生类),函数名,参数,返回值必须全部相同,协变除外,且两个函数必须是虚函数,派生类可以不显示写virtual,但不建议。
  3. 隐藏:两个函数作用域在不同的作用域(一般是基类和派生类),两个函数名相同即可,其它的不关心,父类与派生类的成员变量名相同也构成隐藏。

1.3 纯虚函数和抽象类

  • 纯虚函数:在虚函数后面 加上=0,则这个函数成为纯虚函数。
    示例:
class Car
{
public:
virtual void Drive() = 0;
};
  • 抽象类:包含纯虚函数的类称为抽象类,派生类可以继承抽象类,对纯虚函数进行重写,完成不同的功能。
    示例代码:
class Car
{
public:
virtual void Drive() = 0;
};
class Benz :public Car
{
public:
virtual void Drive()
{
cout << "Benz-舒适" << endl;
}
};

1.4 多态原理

1.4.1 虚函数表指针

看看下面程序在32为程序的运行结果是什么?

  • 示例代码:
class Base
{
public:
	virtual void Func1()
	{
		cout << "Func1()" << endl;
	}

	virtual void Func2()
	{
		cout << "Func2()" << endl;
	}

	void Func3()
	{
		cout << "Func3()" << endl;
	}
protected:
	int _b = 1;
	char _ch = 'x';
};

int main()
{
	Base b;//类内包含一张纯虚函数表,也就是函数指针数组
	cout << sizeof(b) << endl;

	return 0;
}

输出结果为12,因为里面还存在虚函数表指针,指针在32位机器占4字节。
如图:
在这里插入图片描述

1.4.2 原理

构成多态时,运行到指定的对象的虚表中确定对应的虚函数,不再在编译时绑定,称为静态绑定,这是称为动态绑定。

1.4.3 静态绑定与动态绑定

  • 静态绑定:静态绑定是指在程序编译时就确定了方法调用的具体实现。在静态绑定中,编译器根据对象的编译时类型来决定调用哪个方法或访问哪个成员。静态绑定通常发生在方法重载(method overloading)或成员变量访问的场景。
  • 动态绑定:动态绑定是指在程序运行时根据对象的实际类型来决定调用哪个方法或访问哪个成员。动态绑定通常发生在方法重写(method overriding)和多态(polymorphism)的场景中。

1.4.4 虚函数表

  1. 基类对象的虚函数表中存放基类所有的虚函数地址。不同类型的对象具有不同的虚函数表。
  2. 派生类虚函数表里的内容相对复杂,包括基类虚函数地址,派生类重写基类虚函数地址完成覆盖,派生类自己虚函数的地址。
  3. 虚函数表本质是一个存放虚函数指针的指针数组。
  4. 虚函数存在哪?存在代码段,虚函数地址存在虚函数表中。
  5. 虚函数表存在哪?这个C++并没有标准答案,VS存放在代码段(常量区)。

二. 最后

本文深入探讨了C++多态性的核心概念与实现机制,涵盖多态定义、虚函数、协变、析构函数重写、override/final关键字、重载/重写/隐藏对比、纯虚函数与抽象类,以及多态原理如虚函数表指针、静态/动态绑定等,是C++面向对象编程的进阶指南。

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

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

相关文章

《AI大模型应知应会100篇》第60篇:Pinecone 与 Milvus,向量数据库在大模型应用中的作用

第60篇&#xff1a;Pinecone与Milvus&#xff0c;向量数据库在大模型应用中的作用 摘要 本文将系统比较Pinecone与Milvus两大主流向量数据库的技术特点、性能表现和应用场景&#xff0c;提供详细的接入代码和最佳实践&#xff0c;帮助开发者为大模型应用选择并优化向量存储解…

Java学习手册:客户端负载均衡

一、客户端负载均衡的概念 客户端负载均衡是指在客户端应用程序中&#xff0c;根据一定的算法和策略&#xff0c;将请求分发到多个服务实例上。与服务端负载均衡不同&#xff0c;客户端负载均衡不需要通过专门的负载均衡设备或服务&#xff0c;而是直接在客户端进行请求的分发…

Docker私有仓库实战:官方registry镜像实战应用

抱歉抱歉&#xff0c;离职后反而更忙了&#xff0c;拖了好久&#xff0c;从4月拖到现在&#xff0c;在学习企业级方案Harbor之前&#xff0c;我们先学习下官方方案registry&#xff0c;话不多说&#xff0c;详情见下文。 注意&#xff1a;下文省略了基本认证 TLS加密&#xff…

Redis+Caffeine构建高性能二级缓存

大家好&#xff0c;我是摘星。今天为大家带来的是RedisCaffeine构建高性能二级缓存&#xff0c;废话不多说直接开始~ 目录 二级缓存架构的技术背景 1. 基础缓存架构 2. 架构演进动因 3. 二级缓存解决方案 为什么选择本地缓存&#xff1f; 1. 极速访问 2. 减少网络IO 3…

【计算机网络】NAT技术、内网穿透与代理服务器全解析:原理、应用及实践

&#x1f4da; 博主的专栏 &#x1f427; Linux | &#x1f5a5;️ C | &#x1f4ca; 数据结构 | &#x1f4a1;C 算法 | &#x1f152; C 语言 | &#x1f310; 计算机网络 上篇文章&#xff1a;以太网、MAC地址、MTU与ARP协议 下篇文章&#xff1a;五种IO模型与阻…

Python训练打卡Day21

常见的降维算法&#xff1a; # 先运行预处理阶段的代码 import pandas as pd import pandas as pd #用于数据处理和分析&#xff0c;可处理表格数据。 import numpy as np #用于数值计算&#xff0c;提供了高效的数组操作。 import matplotlib.pyplot as plt #用于绘…

node .js 启动基于express框架的后端服务报错解决

问题&#xff1a; node .js 用npm start 启动基于express框架的后端服务报错如下&#xff1a; /c/Program Files/nodejs/npm: line 65: 26880 Segmentation fault "$NODE_EXE" "$NPM_CLI_JS" "$" 原因分析&#xff1a; 遇到 /c/Program F…

并发笔记-信号量(四)

文章目录 背景与动机31.1 信号量&#xff1a;定义 (Semaphores: A Definition)31.2 二元信号量 (用作锁) (Binary Semaphores - Locks)31.3 用于排序的信号量 (Semaphores For Ordering)31.4 生产者/消费者问题 (The Producer/Consumer (Bounded Buffer) Problem)31.5 读写锁 (…

【HTOP 使用指南】:如何理解主从线程?(以 Faster-LIO 为例)

htop 是 Linux 下常用的进程监控工具&#xff0c;它比传统的 top 更友好、更直观&#xff0c;尤其在分析多线程或多进程程序时非常有用。 以下截图就是在运行 Faster-LIO 实时建图时的 htop 状态展示&#xff1a; &#x1f50d; 一、颜色说明 白色&#xff08;或亮色&#xf…

数据同步DataX任务在线演示

数据同步DataX任务在线演示 1. 登录系统 访问系统登录页面&#xff0c;输入账号密码完成身份验证。 2. 环境准备 下载datax安装包&#xff0c;并解压到安装目录 3. 集群创建 点击控制台-多集群管理 计算组件添加DataX 配置DataX引擎,Datax.local.path填写安装目录。 4. …

telnetlib源码深入解析

telnetlib 是 Python 标准库中实现 Telnet 客户端协议的模块&#xff0c;其核心是 Telnet 类。以下从 协议实现、核心代码逻辑 和 关键设计思想 三个维度深入解析其源码。 一、Telnet 协议基础 Telnet 协议基于 明文传输&#xff0c;通过 IAC&#xff08;Interpret As Command…

TAPIP3D:持久3D几何中跟踪任意点

简述 在视频中跟踪一个点&#xff08;比如一个物体的某个特定位置&#xff09;听起来简单&#xff0c;但实际上很复杂&#xff0c;尤其是在3D空间中。传统方法通常在2D图像上跟踪像素&#xff0c;但这忽略了物体的3D几何信息和摄像机的运动&#xff0c;导致跟踪不稳定&#xf…

uniapp 生成海报二维码 (微信小程序)

先下载qrcodenpm install qrcode 调用 community_poster.vue <template><view class"poster-page"><uv-navbar title"物业推广码" placeholder autoBack></uv-navbar><view class"community-info"><text clas…

16.Excel:数据收集

一 使用在线协作工具 简道云。 excel的在线表格协作在国内无法使用&#xff0c;而数据采集最需要在线协作。 二 使用 excel 1.制作表格 在使用excel进行数据采集的时候&#xff0c;会制作表头给填写人&#xff0c;最好还制作一个示例。 1.输入提示 当点击某个单元格的时候&am…

AI系列:智能音箱技术简析

AI系列&#xff1a;智能音箱技术简析 智能音箱工作原理详解&#xff1a;从唤醒到执行的AIPipeline-CSDN博客 挑战真实场景对话——小爱同学背后关键技术深度解析 - 知乎 (zhihu.com) AI音箱的原理&#xff0c;小爱同学、天猫精灵、siri。_小爱同学原理-CSDN博客 智能音箱执行步…

BUUCTF——Ezpop

BUUCTF——Ezpop 进入靶场 给了php代码 <?php //flag is in flag.php //WTF IS THIS? //Learn From https://ctf.ieki.xyz/library/php.html#%E5%8F%8D%E5%BA%8F%E5%88%97%E5%8C%96%E9%AD%94%E6%9C%AF%E6%96%B9%E6%B3%95 //And Crack It! class Modifier {protected $v…

三、Hadoop1.X及其组件的深度剖析

作者&#xff1a;IvanCodes 日期&#xff1a;2025年5月7日 专栏&#xff1a;Hadoop教程 一、Hadoop 1.X 概述 &#xff08;一&#xff09;概念 Hadoop 是 Apache 开发的分布式系统基础架构&#xff0c;用 Java 编写&#xff0c;为集群处理大型数据集提供编程模型&#xff0c;…

PDF2zh插件在zotero中安装并使用

1、首先根据PDF2zh说明文档&#xff0c;安装PDF2zh https://github.com/guaguastandup/zotero-pdf2zh/tree/v2.4.0 我没有使用conda&#xff0c;直接使用pip安装pdf2zh &#xff08;Python版本要求3.10 < version <3.12&#xff09; pip install pdf2zh1.9.6 flask pypd…

springboot3+vue3融合项目实战-大事件文章管理系统-更新用户密码

大致分为这三步 首先在usercontroller中增加updatePwd方法 PatchMapping ("/updatePwd")public Result updatePwd(RequestBody Map<String,String> params){//1.校验参数String oldPwd params.get("old_pwd");String newPwd params.get("n…

C++GO语言socket套接字

目录 01 06-socket-client-server通信过程分析 02 07-socket-server-单次处理 03 08-socket-client 01 09-socket-server-多连接建立 02 10-socket-client多次发送数据 01 -socket-client-server通信过程分析 ### - Server Demo接收一个链接&#xff0c;而且只能发送一次数…