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

目录

前言

多态的原理

动态绑定与静态绑定

虚函数表

小结


前言

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

相关参考文章如下:

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

多态的原理

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

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

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

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

cpp 复制代码
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.虚函数存放在代码段,虚函数的地址存放在虚函数表,虚函数表指针存放在对象中

小结

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

相关推荐
ch.ju几秒前
Java程序设计(第3版)第四章——类的组成
java·开发语言
星轨zb3 分钟前
Spring Data Redis 实战避坑:搞定序列化乱码与 Hash 结构存储
java·redis·spring·lock
艾莉丝努力练剑3 分钟前
【Linux网络】Linux 网络编程:HTTP(四)从手写服务器到生产级 Nginx 与 cpp-httplib 实战
linux·运维·服务器·网络·c++·nginx·http
吴声子夜歌3 分钟前
Java——线程的中断
java·中断
吴声子夜歌3 分钟前
状态机——SpringStateMachine嵌套状态流转
java·状态机·嵌套状态
Jul1en_4 分钟前
【SpringCloud】微服务 Sentinel 详解
java·spring·sentinel
咩咦10 分钟前
C++学习笔记21:日期类加减天数
c++·学习笔记·运算符重载·日期类·operator+·operator+=
闪电悠米11 分钟前
黑马点评短信登录01_session_sms_login
java·spring boot·redis·git·spring·面试
努力努力再努力wz12 分钟前
【QT入门系列】QWidget 六大常用属性详解:windowOpacity、cursor、font、focus、toolTip 与 styleSheet
android·开发语言·数据结构·c++·qt·mysql·算法
少司府13 分钟前
Tools相关:深入浅出学Git
大数据·c++·git·gitee·github·仓库·分支