C++ 多态深度剖析 c++多态如何实现
bigegpt 2024-10-16 07:49 3 浏览
什么是多态?
多态一词最初来源于希腊语,意思是具有多种形式或形态的情形,当然这只是字面意思,它在C++语言中多态有着更广泛的含义。
这要先从对象的类型说起!对象的类型有两种:
想要一起学习C++的可以加群248894430,群内有各种资料满足大家
举个栗子:Derived1类和Derived2类继承Base类
classBase
{};
classDerived1 : publicBase
{};
classDerived2 : publicBase
{};
intmain()
{
Derived1 pd1 = newDerived1;//pd1的静态类型为Derived1,动态类型为Derived1
Base *pb = pd1;//pb的静态类型为Base,动态类型现在为Derived1
Derived2 pd2 = newDerived2;//pd2的静态类型为Derived2,动态类型现在为Derived2
pb = pd2;//pb的静态类型为Base,动态类型现在为Derived2
return0;
}
对象有静态类型,也有动态类型,这就是一种类型的多态。
多态分类
多态有静态多态,也有动态多态,静态多态,比如函数重载,能够在编译器确定应该调用哪个函数;动态多态,比如继承加虚函数的方式(与对象的动态类型紧密联系,后面详解),通过对象调用的虚函数是哪个是在运行时才能确定的。
【静态多态】想要一起学习C++的可以加群248894430,群内有各种资料满足大家
栗子:函数重载
longlongAdd(intleft,intright)
{
returnleft + right;
}
doubleAdd(floatleft,floatright)
{
returnleft + right;
}
intmain()
{
cout<<Add(10,20)<<endl;//语句一
cout<<Add(12.34f,43.12f)<<endl;//语句二
return0;
}
编译器在编译期间完成的,编译器根据函数实参的类型(可能会进行隐式类型转换),即可推断出要调用哪个函数,如果有对应的函数就调用该函数,否则出现编译错误。
【动态多态】
进入动态多态前,先看一个普通继承的栗子:
classPerson
{
public:
voidGoToWashRoom()
{
cout<<"Person-->?"<<endl;
}
};
classMan : publicPerson
{
public:
voidGoToWashRoom()
{
cout<<"Man-->Please Left"<<endl;
}
};
classWoman : publicPerson
{
public:
voidGoToWashRoom()
{
cout<<"Woman-->Please Right"<<endl;
}
};
intmain()
{
Person per, *pp;
Man man, *pm;
Woman woman, *pw;
pp = &per;
pm = &man;
pw = &woman;
//第一组 //这些都是毫无疑问的
per.GoToWashRoom();//调用基类Person类的函数
pp->GoToWashRoom();//调用基类Person类的函数
man.GoToWashRoom();//调用派生类Man类的函数
pm->GoToWashRoom();//调用派生类Man类的函数
woman.GoToWashRoom();//调用派生类Woman类的函数
pw->GoToWashRoom();//调用派生类Woman类的函数
//第二组
pp = &man;
pp->GoToWashRoom();//调用基类Person类的函数
pp = &woman;
pp->GoToWashRoom();//调用基类Person类的函数
return0;
}
运行结果:
第一组毫无疑问,通过本类对象和本类对象的指针就是调用本类的函数;第二组中先让基类指针指向子类对象,然后调用该函数,调用的是子类的,后让基类指针指向另一个子类对象,调用的是子类的函数。这是因为p的类型是一个基类的指针类型,那么在p看来,它指向的就是一个基类对象,所以调用了基类函数。就像一个int型的指针,不论它指向哪,读出来的都是一个整型(在没有崩溃的前提下),即使将它指向一个float。再来对比着看下一个栗子。
栗子:继承+虚函数
classPerson
{
public:
virtual voidGoToWashRoom() = 0;
};
classMan : publicPerson
{
public:
virtual voidGoToWashRoom()
{
cout<<"Man-->Please Left"<<endl;
}
};
classWoman : publicPerson
{
public:
virtual voidGoToWashRoom()
{
cout<<"Woman-->Please Right"<<endl;
}
};
intmain()
{
for(inti = 0;i < 10;i++)
{
Person *p;
if(i&0x01)
p = newMan;
else
p = newWoman;
p->GoToWashRoom();
deletep;
sleep(1);
}
return0;
}
运行结果:
就像上边这个栗子所演示的那样,通过重写虚函数(不再是普通的成员函数,是虚函数!),实现了动态绑定,即在程序执行期间(非编译期)判断所引用对象的实际类型,根据其实际类型调用相应的方法。使用virtual关键字修饰函数时,指明该函数为虚函数(在栗子中为纯虚函数),派生类需要重新实现,编译器将实现动态绑定。在上边栗子中,当指针p指向Man类的对象时,调用了Man类自己的函数,p指向Woman类对象时,调用了Woman类字几的函数。
今天的重点来了,就是要分析这个动态绑定实现的原理(以下测试在VS2013环境下进行):
为了方便调试,我将程序修改如下:
classPerson
{
public:
voidGoToWashRoom()
{};
};
classMan : publicPerson
{
public:
voidGoToWashRoom()
{
cout << "Man-->Please Left" << endl;
}
};
intmain()
{
cout << sizeof(Person) << endl;
cout << sizeof(Person) << endl;
return0;
}
先求一下两个普通的继承类的大小,在这里为空类,没有包含成员变量,所以为1,表示占位:
假如基类中包含一个int型变量,那么这里的大小都会是4。这不是今天的重点,不再叙说。主要是想看看普通空类的大小。
再改一下程序,在基类的成员函数前加virtual关键字,将这个函数变为虚函数。
classPerson
{
public:
virtual voidGoToWashRoom()
{};
};
其它部分代码不变,运行结果:
大小变成了4.所以这个类里面肯定是有什么东西的。
在main中加入以下代码:
Man man;
Person *p = &man;
p->GoToWashRoom();
查看监视窗口:
man对象里存在了一个指针,而且是void**类型的,那么这个指针指向哪呢?可以在内存中查看一下这个地址所在内存中的内容。
是一个可能是地址的东西,然后下边是一排的 00 00 00 00,貌似相当于NULL。继续看一下这个类似于地址的东西里面又是什么:
嗯,看不懂,不要紧,把这个数字记下来:0x01 27 13 de
继续运行程序,转到反汇编:
这两句取到man的首地址然后通过eax寄存器赋值给p,这样p就指向了man对象。目前对象模型是这样的:
同时,_vfptr的值为0x01 27 13 de。接下来就要准备调用函数了。
一句句分析:
01276036 mov eax,dword ptr [p] //从p所指位置取4个字节内容放到eax,其实就是取的man对象的地址:0x008BFC1C
01276039 mov edx,dword ptr [eax] //从eax所指位置取4个字节内容放到edx,就是我们之前看到的不明变量_vfptr的值:0x0127dc80
0127603B mov esi,esp //这句先不用管,esp是栈顶指针
0127603D mov ecx,dword ptr [p] //将对象地址给ecx也保存了一份,此时ecx和eax放的都是对象地址(其实这句就是调用函数之前通过ecx传递this指针)
01276040 mov eax,dword ptr [edx] //取对象前四个字节给eax,eax和edx都变成了_vfptr的值。
01276042 call eax //调用函数,函数地址在eax中,说明_vfptr指向的内容是函数的地址
01276044 cmp esi,esp
01276046 call __RTC_CheckEsp (01271334h)
到call这条语句跟进去看一下:
这下明白了吧,其实_vfptr存放的内容就是存放函数地址的地址,即_vfptr指向函数地址所在的内存空间,如图:
分析暂告一段落。我们知道了man对象中维护了一个虚表指针,虚表中存放着虚函数的地址。基于这个虚表指针,实现动态绑定,才可以用基类指针调用了Man类中的虚函数。因为指针p看到的是虚表的指针,它调用的虚函数是从虚表中查找的。如果基类中有多个虚函数的话,那么虚表中也会依次按基类中虚函数定义顺序存放虚函数的地址,并以0x 00 00 00 00 结尾。再如果子类中有自己定义的新的虚函数,那么会排在虚函数表的后边。在调用虚函数时由编译器自动计算偏移取得相应的虚函数地址。
看看是如何构造子类对象的:
classPerson
{
public:
Person()
{
cout << "Person()" << endl;
}
~Person()
{
cout << "~Person()" << endl;
}
virtual voidGoToWashRoom()
{};
};
classMan : publicPerson
{
public:
Man()
{
cout << "Man()" << endl;
}
~Man()
{
cout << "~Man()" << endl;
}
voidGoToWashRoom()
{
cout << "Man-->Please Left" << endl;
}
};
intmain()
{
Man man;
return0;
}
运行结果:
先调用基类构造函数,虚表指针先指向基类虚表,然后调用子类构造函数,这时候子类对象的虚表指针就指向了子类自己的虚表。这才是动态绑定实现原理。详细内容可以查看反汇编,这里不再写了。
再举个栗子看看虚表指针在对象的首部还是尾部,又或者是在中间某个地方存放:
classPerson
{
public:
virtual voidGoToWashRoom()
{};
public:
inti1;
inti2;
inti3;
};
classMan : publicPerson
{
voidGoToWashRoom()
{
cout << "Man-->Please Left" << endl;
}
};
intmain()
{
Man man;
man.i1 = 0x01;
man.i2 = 0x02;
man.i3 = 0x03;
return0;
}
查看内存中:&man
可以看出,虚表指针位于对象的首部。想要一起学习C++的可以加群248894430,群内有各种资料满足大家
【动态绑定条件】
必须是虚函数
通过基类类型的引用或者指针调用
总结
派生类重写基类的虚函数实现多态,要求函数名、参数列表、返回值完全相同。(协变除外)
基类中定义了虚函数,在派生类中该函数始终保持虚函数的特性
只有类的成员函数才能定义为虚函数,静态成员函数不能定义为虚函数
如果在类外定义虚函数,只能在声明函数时加virtual关键字,定义时不用加
构造函数不能定义为虚函数,虽然可以将operator=定义为虚函数,但最好不要这么做,使用时容 易混淆
不要在构造函数和析构函数中调用虚函数,在构造函数和析构函数中,对象是不完整的,可能会 出现未定义的行为
最好将基类的析构函数声明为虚函数。(析构函数比较特殊,因为派生类的析构函数跟基类的析构 函数名称不一样,但是构成覆盖,这里编译器做了特殊处理)
虚表是所有类对象实例共用的
//协变,也可以构成重写(覆盖),但返回值是该类类型的指针或引用
classBase
{
virtual Base *FunTest()
{
//do something
}
};
classDerived : publicBase
{
Derived *FunTest()
{
//do something
}
};
容易混淆的点:
想要一起学习C++的可以加群248894430,群内有各种资料满足大家
相关推荐
- 无畏契约手游测试资格获取方法,安卓IOS下载教程
-
《无畏契约:源能行动》是拳头游戏与腾讯光子工作室联合开发的《无畏契约》正版手游,延续了端游的5v5战术射击核心玩法,并针对移动端进行了操作优化。游戏以快节奏的爆破模式为核心,融合角色技能系统、经济策略...
- 微软正在测试重新设计的Office图标 但您现在可以提前下载重制版本
-
今年4月,有消息称微软正在征求用户对一组Office图标7年来首次重制版的看法(上一次重制是在2018年末)。现在,有人决定自己动手,制作了一套微软的高分辨率图标包与用户共享以获得反馈。Reddi...
- AB Download Manager:一款可以替代IDM的开源桌面下载管理器
-
软件介绍IDM下载器大家应该多少都知道一点,如果不知道的话只能自行百度了,但是IDM本身是需要付费的,而今天推荐的这款软件,在下载方面是和IDM差不多的,大概有90%的相似度,感兴趣的朋友可以体验一下...
- 《夺宝奇兵》PS5光盘仅20G:其余需联网下载
-
来源:游民星空【《夺宝奇兵》PS5光盘仅20G:其余需联网下载】据游戏测试账号“DoesItPlay1”在推特发布动态表示,《夺宝奇兵:古老之圈》PS5实体光盘只存储了20GB的游戏数据,其余内容需要...
- 薇姐聊诗词7:诗词创作韵部查询及检测工具
-
薇姐聊诗词7:诗词创作韵部查询及检测工具。·1、诗词创作中所用韵脚哪里找?平水韵:106部,分平声30部、上声29部、去声30部、入声17部,反映中古汉语语音体系。新韵:(中华新韵)14部,以普通话为...
- 阿里云国际站:怎样模拟高并发测试场景?
-
本文由【云老大】TG@yunlaoda360撰写一、使用JMeter安装JMeter:从JMeter官网下载并安装JMeter。创建测试计划:打开JMeter,创建一个新的测试计划。添加线程组...
- Android Studio 新增 AI 驱动的测试和更智能的崩溃诊断功能
-
随着GoogleI/O2025大会的落幕,值得注意的是,谷歌在AndroidStudio中引入了几项新功能,旨在改善Android应用程序的开发流程。最新版本集成了更先进的AI工...
- 如何在本地测试PHP源码的网站
-
通常,我们测试自建网站或从网上获取的PHP源码时,若直接上传到服务器,出错后再修改会很麻烦,因此一般会选择先在本地电脑上进行测试。1、先下载喜欢的源码,很多网站提供下载,如源码论坛等。这些源码是现成...
- 显卡性能测试工具3DMark06的应用教程
-
显卡作为计算机的重要组成部分,也是主要的输出设备。在计算机系统中,图形处理性能的瓶颈往往在于显卡。若要评估显卡性能,用户可以借助专业的检测工具3DMark,判断显卡是否能满足当前需求,或者是否需要...
- Downie4 安装教程(轻松获取视频素材)
-
效果一、准备工作下载软件链接:http://www.macfxb.cn二、开始安装1、双击运行软件,将其从左侧拖入右侧文件夹中,等待安装完毕2、应用程序显示软件图标,表示安装成功三、运行测试1、打开软...
- 如何使用瑞星杀毒软件的网速测试功能
-
下面为大家介绍瑞星杀毒软件的网速测试功能。1、打开安全工具,找到网速测试,点击下载后开启。2、打开网速测试页面,点击开始测试按钮。3、测试结束后,你就能知晓自己的网速了。(9744667)...
- 阿里云国际站:如何测试服务器真实带宽?
-
本文由【云老大】TG@yunlaoda360撰写基于命令行工具测试iperf/iperf3:服务器端:在服务器上安装iperf后,运行iperf-s或iperf3-s启动服务端,...
- CentOS Docker 安装
-
Docker支持以下的64位CentOS版本:CentOS9(stream)更高版本...必须启用centos-extras仓库,该仓库默认启用,如果您禁用了它,需要重新启用。使用官...
- Fast YOLO:用于实时嵌入式目标检测(附论文下载)
-
关注并星标从此不迷路计算机视觉研究院公众号ID|ComputerVisionGzq计算机视觉研究院专栏作者:Edison_G目标检测被认为是计算机视觉领域中最具挑战性的问题之一,因为它涉及场景中对象分...
- aigc检测报告与查重监测报告
-
哈喽学妹学弟们!最近是不是都在忙着写论文呢?记得当初我写论文的时候,也被AIGC检测报告和查重监测报告搞得晕头转向。不过经过我的一番摸索,终于搞清楚了它们之间的区别和联系。来来来,学姐今天就来给你们传...
- 一周热门
- 最近发表
- 标签列表
-
- mybatiscollection (79)
- mqtt服务器 (88)
- keyerror (78)
- c#map (65)
- resize函数 (64)
- xftp6 (83)
- bt搜索 (75)
- c#var (76)
- mybatis大于等于 (64)
- xcode-select (66)
- mysql授权 (74)
- 下载测试 (70)
- httperror403.14-forbidden (63)
- logstashinput (65)
- hadoop端口 (65)
- dockernetworkconnect (63)
- esxi7 (63)
- vue阻止冒泡 (67)
- oracle时间戳转换日期 (64)
- jquery跨域 (68)
- php写入文件 (73)
- kafkatools (66)
- mysql导出数据库 (66)
- jquery鼠标移入移出 (71)
- 取小数点后两位的函数 (73)