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

C++从入门到入土(八)——多态的原理

目录

前言

多态的原理

动态绑定与静态绑定

虚函数表

小结


前言

在前面的文章中,我们介绍了C++三大特性之一的多态,我们主要介绍了多态的构成条件,但是对于多态的原理我们探讨的是不够深入的,下面这这一篇文章,我们将着重介绍C++多态的实现原理。

相关参考文章如下:

C++从入门到入土(七)——多态

多态的原理

我们在前面的文章中了解到多态的实现条件有以下两个:

1.必须是基类的指针或引用调用虚函数

2.被调用的函数必须是虚函数,并且完成了重写

看到上面两个条件,我们不禁会发出疑问,为什么必须是通过基类的指针或引用调用虚函数呢?为什么不能通过其他条件构成多态呢?那么我们通过下面代码来讨论这两个问题:

class Base
{
public:
	virtual void Func()
	{
		cout << "基类调用" << endl;
	}
protected:
	int _base;
};

class A:public Base
{
public:
	virtual void Func()
	{
		cout << "派生类调用" << endl;
	}
private:
	int _a;
};

void Print(Base& a)
{
	a.Func();
}

int main()
{
	Base b;
	Print(b);

	A a;
	Print(a);
	return 0;
}

首先我们看到上述一段简单的代码,其运行结果如下所示:

我们进入调试观察一下它的内部:

我们可以看到,相比于没有实现多态的类,其底层多了一个_vfptr的对象,那这个对象是什么呢?直接讲结论:这个对象是虚函数表(v表示virtual,f表示function,ptr表示指针),其本质是一个指针数组。

那么我们此时就可以猜测,是不是因为虚函数表的出现导致多态行为的发生呢?我们再仔细观察一下会发现,派生类的虚函数表保存在基类base之下,但是他们的地址是不同的,那么虚函数表中存储的是什么呢?实际上虚函数表中存的是虚函数的地址。当我们满足多态的条件后,我们在编译时不再通过调用对象来确定函数的地址,而是运行时到指向对象的虚函数表中确定虚函数的地址,这样就实现了基类的指针或引用调用不同函数的目的

动态绑定与静态绑定

我们在粗略理解了多态的原理后,接下来我们引入动态绑定与静态绑定的概念帮助我们更加深入地理解多态地原理.。

首先我们要理解什么是动态与静态绑定。

静态绑定:在编译时就确定函数的地址就叫静态绑定,换句话说就是不满足多态的函数调用,例如:函数重载、模板等。

动态绑定:在运行时通过虚函数表来确定函数的地址就叫动态绑定,例如:多态。

所以可以这么说,动态绑定是多态的特点之一。

虚函数表

还是以上面的例子,我们发现,派生类和基类在实例化的过程中分别产生了不同的虚函数表,即使派生类继承了基类的对象,但是派生类的虚函数表中并没有保存基类虚函数的地址,我们将虚函数表的地址输入到内存窗口中查看:

 我们可以看到,基类和派生类的虚函数表中保存的地址是不同的,不同的类的虚表也是不同的。

我们给派生类再加一个虚函数,进入调试,观察一下:

当我们给基类添加了一个虚函数后发现,虚函数表中又存储了一个地址,但是对于基类而言只有一个虚函数的地址,那么我们可以得出下面的结论:

派生类的虚函数表中包含<1.>基类虚函数的地址 <2.>派生类重写的虚函数的地址的覆盖 <3.>自己的虚函数的地址

综上所述,对于虚函数表的结论如下:

1.基类对象的虚函数表存放基类所有虚函数的地址,同类型的对象共用一张虚函数表,不同类型的对象各有自己的虚函数表,即:基类和派生类各有自己的虚函数表

2.虚函数表中包含:基类虚函数的地址;  派生类重写的虚函数的地址的覆盖; 派生类自己的虚函数

3.虚函数表本质是一个函数指针数组,存放虚函数的地址

4.虚函数存放在代码段,虚函数的地址存放在虚函数表,虚函数表指针存放在对象中

小结

本篇文章我们介绍了多态的实现原理,动态绑定以及虚函数表,通过本篇博客的阅读相信您对多态的认识会更加清楚,如果本篇文章对您有所帮助的话希望能够点赞、关注加转发,您的支持就是我创作的最大动力。  


http://www.kler.cn/a/590353.html

相关文章:

  • 冒泡排序:古老算法中的智慧启示
  • 「Java EE开发指南」如何用MyEclipse构建一个Web项目?(二)
  • Zabbix7.0+DeepSeek大模型实现人工智能告警分析
  • 鸿蒙路由 HMRouter 配置及使用 二
  • WebSocket与MQTT协议深度对比:选择合适的通信协议
  • 如何用Python批量将CSV文件编码转换为UTF-8并转为Excel格式?
  • 技术与情感交织的一生 (一)
  • 现代密码学 | 具有数字签名功能的安全方案
  • Spring MVC 全面解析:架构、流程与核心组件(详细)
  • spring bean的生命周期和循环依赖
  • 【零基础入门unity游戏开发——unity2D篇】2D射线和范围检测之Physics2D Raycast、OverlapCircle、OverlapBox
  • golang函数与方法的区别
  • K8S快速部署
  • 新闻发布时间抽取分析
  • LinkedList和链表
  • 【MySQL】从零开始:掌握MySQL数据库的核心概念
  • containerd 拉取镜像的工具以及优劣
  • 系统架构设计师—案例分析—架构评估
  • LLM论文笔记 24: A Theory for Length Generalization in Learning to Reason
  • QT非UI设计器生成界面的国际化