C++中通过虚函数实现多态的原理

devtools/2025/3/15 10:57:46/

C++中通过虚函数实现多态的原理

我们都知道C++是通过虚函数实现多态的,那么其中的原理是什么呢?

在C++中,多态性是一种重要的特性,它允许通过基类指针或引用来调用派生类中的函数。多态性主要分为两种:编译时多态(主要通过函数重载和模板实现)和运行时多态(主要通过虚函数实现)。C++中通过虚函数实现的多态主要是运行时多态(动态多态),其原理主要基于虚函数表(vtable)和虚函数指针(vptr)。

1. 虚函数的定义

虚函数是在基类中使用virtual关键字声明的成员函数。例如:

class Base 
{
public:virtual void show() {cout << "Base::show" << endl;}
};

当一个类的成员函数被声明为虚函数后,C++运行时系统会为该类及其派生类提供动态绑定的机制。

2. 虚函数表(vtable)

虚函数表是一个由编译器生成的数组,其中存储了类中所有虚函数的地址。每个包含虚函数的类都有一个唯一的虚函数表。例如:

class Base 
{
public:virtual void show() {cout << "Base::show" << endl;}virtual void display() {cout << "Base::display" << endl;}
};

对于Base类,编译器会为其生成一个虚函数表,其中包含showdisplay的地址。

3. 虚函数指针(vptr)

每个包含虚函数的类的对象都会有一个隐藏的成员变量——虚函数指针(vptr)。这个指针指向该对象所属类的虚函数表。当对象被创建时,构造函数会初始化这个指针,使其指向正确的虚函数表。

4. 多态的实现原理

当通过基类指针或引用调用虚函数时,C++运行时系统会通过对象的vptr找到对应的vtable,然后通过vtable找到正确的虚函数地址并调用。这个过程称为动态绑定或晚期绑定。

#include <iostream>
using namespace std;class Base {
public:virtual void show() {cout << "Base::show" << endl;}
};class Derived : public Base {
public:void show() override {cout << "Derived::show" << endl;}
};int main() {Base* ptr;Base base;Derived derived;ptr = &base;ptr->show();  // 输出:Base::showptr = &derived;ptr->show();  // 输出:Derived::showreturn 0;
}

解释:

  1. Base base对象:
    • base对象的vptr指向Base类的vtable。
    • Base类的vtable中存储了Base::show的地址。
    • 当调用ptr->show()时,运行时通过ptr对象的vptr找到Base类的vtable,并调用Base::show
  2. Derived derived对象:
    • derived对象的vptr指向Derived类的vtable。
    • Derived类的vtable中存储了Derived::show的地址(覆盖了Base::show)。
    • 当调用ptr->show()时,运行时通过ptr对象的vptr找到Derived类的vtable,并调用Derived::show

5. 虚函数表的存储结构

假设Base类和Derived类的虚函数表存储如下:

  • Base类的vtable:
    • Base::show
    • Base::display(如果有其他虚函数)
  • Derived类的vtable:
    • Derived::show(覆盖了Base::show
    • Base::display(如果没有覆盖,仍然指向基类的虚函数)

6. 多态的条件

要实现通过虚函数的多态,必须满足以下条件:

  1. 继承关系: 派生类继承自基类。
  2. 虚函数: 基类中必须有虚函数。
  3. 覆盖: 派生类覆盖了基类的虚函数。
  4. 使用基类指针或引用: 通过基类指针或引用调用虚函数。

7. 总结

通过虚函数实现多态的核心是虚函数表(vtable)和虚函数指针(vptr)。vtable存储了类中虚函数的地址,vptr指向对象所属类的vtable。运行时通过vptr找到正确的vtable,再通过vtable找到正确的虚函数地址并调用,从而实现动态绑定。

这种机制使得程序能够在运行时根据对象的实际类型调用对应的虚函数,而不是根据指针或引用的类型,从而实现了多态。


http://www.ppmy.cn/devtools/167272.html

相关文章

C++类对象创建全解析:从构造函数到内存管理

目录 对象的创建 对象的创建规则 对象的数据成员初始化 对象所占空间大小 总结 指针数据成员 对象的创建 在之前的 Computer 类中&#xff0c;通过自定义的公共成员函数 setBrand 和 setPrice 实现了对数据成员的初始化。实际上&#xff0c;C 为类提供了一种特殊的成员函…

mac安装python没有环境变量怎么办?zsh: command not found: python

在mac电脑上,下载Python安装包进行安装之后,在终端中,输入python提示: zsh: command not found: python 一、原因分析 首先,这个问题不是因为python没有安装成功的原因,是因为python安装的时候,没有为我们添加环境变量导致的,所以我们只需要,在.zshrc配置文件中加上环…

研究整除的性质——最大公约数(GCD)和最小公倍数(LCM)

最大公约数&#xff08;GCD&#xff09;和最小公倍数&#xff08;Least Common Multiple&#xff0c;LCM&#xff09;研究整除的性质&#xff0c;非常古老&#xff0c;在2000多年前就得到了很好的研究。由于简单易懂&#xff0c;有较广泛的应用&#xff0c;它们是竞赛中频繁出现…

【AI论文】MM-Eureka:基于规则的大规模强化学习探索视觉“啊哈”时刻

摘要&#xff1a;我们提出了MM-Eureka&#xff0c;这是一个多模态推理模型&#xff0c;成功地将基于规则的大规模强化学习&#xff08;RL&#xff09;扩展到多模态推理领域。虽然基于规则的RL在提升大型语言模型&#xff08;LLMs&#xff09;在文本领域的推理能力方面已经取得了…

Qt项目中集成第三方模块的.pri文件

对于功能模块较多的Qt项目&#xff0c;使用pri文件管理模块文件&#xff0c;降低工程复杂度&#xff0c;提高软件模块的封装性和重用性。 一、.pro与.pri 对于模块化编程&#xff0c;Qt提供了pro和pri&#xff0c;pro管理项目&#xff0c;pri管理模块。 .pro 文件是Qt项目的…

TDengine 使用最佳实践

简介 阅读本文档需要具备的基础知识&#xff1a; Linux系统的基础知识&#xff0c;及基本命令网络基础知识&#xff1a;TCP/UDP、http、RESTful、域名解析、FQDN/hostname、hosts、防火墙、四层/七层负载均衡 本文档的阅读对象有&#xff1a;架构师、研发工程师&#xff0c;…

Python 推导式详解

Python 推导式详解 Python 推导式&#xff08;Comprehension&#xff09;是一种简洁高效的语法结构&#xff0c;用于快速生成列表、字典、集合等数据结构。它通过一行代码替代传统的多行循环&#xff0c;使代码更简洁易读。以下是常见的推导式类型及用法详解&#xff1a; 一、…

C#设计模式Demo——MVC

设计模式Demo——MVC 1.View1.1页面示例1.2View代码1.3修改界面以及代码 2.Model3.Controller4.数据结构5枚举类型6.工具类6.1缓存信息6.2扩展类. 文件结构图 1.View 1.1页面示例 1.2View代码 using System; using System.Data; using System.Windows.Forms; using MVC模式…