C++: 多态

目录

一、多态的概念

二、多态的定义及实现

2.1虚函数

2.2虚函数的重写

2.3多态的构成条件

2.4虚函数重写的两个例外

1.协变

2.析构函数的重写

2.5虚函数重写的实质

2.6override 和 final(C++11)

1.final

2.override

2.7重载、覆盖(重写)和隐藏(重定义)的区别

三、抽象类

3.1概念

3.2接口继承和实现继承

四、多态的原理

4.1虚函数表

4.2多态的原理

4.3静态绑定和动态绑定

五、单继承和多继承关系中的虚函数表

5.1单继承

5.2多继承中的虚函数表

5.3菱形继承、菱形虚拟继承


一、多态的概念

      完成某个行为时,不同对象会产生不同状态。

二、多态的定义及实现

2.1虚函数

虚函数:即被virtual修饰的类成员函数称为虚函数。

class A
{virtual void Print(){cout<<A::Print()<<endl;}
};

2.2虚函数的重写

虚函数的重写(覆盖):派生类有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称派生类的虚函数重写了基类的虚函数。

大家看到这可能会想到前面继承学的隐藏,回顾一下隐藏的概念。只要父类和子类有共同的成员名字,就构成隐藏,这不需要父类和子类的函数参数也相同。虚函数的重写比隐藏的定义更严格,两个分别在父类和子类函数的关系,函数名相同,不是重写,就是隐藏。如果满足重写,那就是重写。

class A
{
public:virtual void Print(){cout<<A::Print()<<endl;}
};class B: public A
{public:virtual void Print(){cout<<B::Print()<<endl;}
};

上述代码就展示了虚函数的重写,注意虽然子类可以不加virtual 也能和父类构成重写,但这种写法不太规范,不建议使用。

2.3多态的构成条件

       多态是在不同的继承关系的类对象,去调用同一函数,产生了不同的行为。

在继承中要构成多态有两个条件:

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

2.被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写

class A
{
public:virtual void Print(){cout<<A:Print()<<endl;}
};class B: public A
{
public:virtual void Print(){cout<<B:Print()<<endl;}
};void Func(A& a)
{a.Print();
}int main()
{A a1;Func(a1);B b1;Func(b1);return 0;
}

传给Func的是基类的引用,传的是基类,调用的就是基类的函数,传的是派生类,那么调用的就是派生类的函数。非常方便。Func的函数是灵魂,它的参数是基类的引用或者指针。

2.4虚函数重写的两个例外

1.协变

(基类与派生类虚函数返回类型不同)

派生类重写基类虚函数时,与基类虚函数返回类型不同。基类虚函数返回基类对象的指针或引用,派生类虚函数返回派生类对象的指针或引用。称为协变。

2.析构函数的重写

如果基类函数的析构函数是虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字,都与基类的析构函数构成重写,虽然此时基类与派生类析构函数名字不同,但这里是编译器对析构函数名称进行了统一处理,处理成了destructor,所以可以看成是相同的。这时候,只有delete调用析构函数的时候,才能构成多态。

2.5虚函数重写的实质

虚函数重写,继承了接口,重写的是具体的实现。

2.6override 和 final(C++11)

这两个关键字是用来帮助用户检测某个函数是否重写的。

1.final

修饰虚函数,表示该虚函数不能再被重写

class A
{
public:virtual void Func() final{}};

2.override

检查派生类虚函数是否重写了基类某个虚函数,如果没有重写编译会报错

class A
{
virtual void Func()
{cout<<A::Func()<<endl;
}};class B: public A
{virtual void Func() override{cout<<B::Func()<<endl;}
};

2.7重载、覆盖(重写)和隐藏(重定义)的区别

重载

a.两个函数在同一个作用域

b.函数名相同,参数类型、顺序或个数不同

重写(覆盖)

a.两个函数分别在基类和派生类作用域

b.函数名,参数和返回值都必须相同(协变例外)

c.两个函数必须是虚函数

重定义(隐藏)

a.两个函数分别在基类和派生类作用域

b.两个函数的函数名相同

c.基类和派生类的函数名相同,不构成重写就是重定义

三、抽象类

3.1概念

在虚函数的后面写个=0,那么这个虚函数就是纯虚函数。包含纯虚函数的类叫抽象类。纯虚函数不能实例化对象,继承以后的纯虚函数也不能实例化对象,它规定必须重写这个函数才能使用。纯虚函数体现了接口继承,因为虚函数的重写本就是继承了接口,重写了实现。这里纯虚函数重写主要就是继承了接口。

3.2接口继承和实现继承

普通函数的继承,派生类继承了基类的函数,继承的是实现。

虚函数继承是一种接口继承,继承的是基类的接口,就是参数的列表,目的是为了重写,达成多态。达成同一个接口但能有不同的反应。所以如果不实现多态,就不要写虚函数。

四、多态的原理

4.1虚函数表

写个代码,基类为Person,有两个虚函数分别是Print()和Func2()和一个普通函数Func3(),派生类为Student,继承了Person,并且重写了Person里的Print()。

namespace ting 
{class Person{public:virtual void Print(){cout << "买票全价" << endl;}virtual void Func2(){cout<<"Person::Func2()" << endl;}void Func3(){cout << "Person::Func3()" << endl;}};class Student: public Person{public:virtual void Print(){cout << "买票85折" << endl;}private:int a = 18;};void test1(){Person p1;Student s1;}
}

测试代码主要是对test1()的调用,这里暂时不写了。通过对上述代码调试,监视窗口可以看到s1和p1里都有一个_vfptr放在对象的前面(如下图所示),对象中的这个指针我们叫做虚函数表指针(v即virtual,f即function,ptr即指针)

一个含有虚函数的类中至少要有一个虚函数表指针这个指针指向一个虚函数表,这个虚函数表里存储虚函数的地址,虚函数表也叫虚表。

通过观察,我们得到以下结论:

1.派生类对象,上述s1中,包含的内容有两部分。一是它从父类继承的成员,比如虚表指针,第二部分是它自己的成员。

2.基类对象和派生类对象不一样,这里Print()在派生类进行了重写,所以在s1里的虚表里原来存Person::virtual void Print()这个函数的地址,现在存Student::virtual void Print()函数的地址。完成了覆盖。重写是语法层面上的,覆盖是原理层上的叫法。

3.Func2继承下来了,它是虚函数,所以放在虚表中,Func3继承了,不过它不是虚表,所以没放在虚表里。

4.虚函数表本质是一个指针数组,一般这个数组的最后放了nullptr.

5.派生类的虚表生成:1,先将基类的虚表拷贝一份到派生类虚表中 2,如果派生类重写了基类的某个虚函数,则用这个虚函数覆盖虚表内的虚函数。 3,派生类自己新增加的虚函数按其在派生类的声明次序,增加到派生类虚表的最后。

6.对象中存的是虚表指针,虚表指针中存的是虚表,虚表中存的是虚函数指针,虚函数指针指向虚函数,虚函数和普通函数一样存在代码段。虚表在vs下存在代码段。

4.2多态的原理

拿出上面的代码分析。

class A
{
public:virtual void Print(){cout<<A:Print()<<endl;}
};class B: public A
{
public:virtual void Print(){cout<<B:Print()<<endl;}
};void Func(A& a)
{a.Print();
}int main()
{A a1;Func(a1);B b1;Func(b1);return 0;
}

在Func调用基类的对象a1时,Func从传来的引用,基类的对象a1中去找虚函数指针,找虚表,然后找到要调用的函数。如果是派生类的对象b1,就从b1中去找虚函数指针,找虚表,找到相应的地址和指向的函数,从而实现了多态,即相同的接口,传的对象不同,实现不同的行为。

多态的函数调用,并不是在编译时确定的,而是在运行时在对象中去找的。

4.3静态绑定和动态绑定

静态绑定又称为前期绑定,在程序编译时就确定了程序的行为,也成为静态多态。比如:函数重载。

动态绑定又称为晚期绑定,在运行期间,根据具体拿到的类型,确定程序的具体行为,调用具体的函数,称为动态多态。

五、单继承和多继承关系中的虚函数表

看到这一节,即将进入王炸阶段。复杂程度可以把初学者按在地上狠狠摩擦。(手动呲牙)

这里主要关注的是派生类的虚函数表。

5.1单继承

如何打印派生类B的对象的虚表?

namespace ting
{class A{public:virtual void Func1() {};virtual void Func2() {};virtual void Func3() {};virtual void Func4() {};};class B: public A{public:virtual void Func1(){cout << "B:Func1()" << endl;}};void test2(){A a1;B b1;}
}

具体过程在下面代码中注释了,这里主要讲几个要点

1.为了获取虚表地址,首先获取a1地址,a1是整个对象的地址,我们要取虚表的地址,虚表地址在a1的前面4个或8个字节(32位是4个字节,64位是8个字节)。可以通过强制类型转换。再解引用。

不过这里采用先转为VFPTR的二级指针,VEPTR*是虚表指针,VFPTR**是指向虚表的指针,通过先转化位二级指针,再解引用,转化为虚表指针。这样比较安全,一般情况下,直接强制转换成VFPTR*可能不太准确。

2.得到虚表指针的地址,就可以访问这个虚表指针数组,这个虚表指针数组里存的都是虚表地址。

由于前面提到了虚表指针数组的最后一个元素是nullptr,所以可以通过这个条件来遍历整个指针数组,打印这个虚表里存的地址,打印地址的占位符是%p,64位。还需要把这个VFPTR强制转换为void*,才可以进行打印。

3.取出虚表指针数组中存储的虚函数指针,用这个指针来调用相应的函数,在函数内部进行区别。就能看到地址对应的函数和虚表中存储的全部虚函数地址了!

namespace ting
{class A{public:virtual void Func1() { cout << "A:Func1()" << endl; };virtual void Func2() { cout << "A:Func2()" << endl; };virtual void Func3() { cout << "A:Func3()" << endl; };virtual void Func4() { cout << "A:Func4()" << endl; };};class B: public A{public:virtual void Func1(){cout << "B:Func1()" << endl;}};typedef void(*VFPTR) ();//把这个函数指针,重命名为VFPTRvoid PrintVFTable(VFPTR vtable[]){cout << " 虚表地址:" << vtable << endl;for (int i = 0;vtable[i] != nullptr; ++i){printf("0x%p->", (void*)vtable[i]);//这里是打印地址,用printfVFPTR f = vtable[i];//取上面那个函数地址f();//用指针调用函数,函数名就是函数地址}}void test2(){A a1;B b1;//取虚表地址并强制转换称VFPTR类型VFPTR* vtable_a1 = *(VFPTR**)(&a1);PrintVFTable(vtable_a1);VFPTR* vtable_b1 = *(VFPTR**)(&b1);PrintVFTable(vtable_b1);}
}

这里可以很直观的看到,由于Func1()被重写了,所以这里的派生类对象b1里的Func1(),是由重写后的虚函数地址,覆盖了原来的地址。

5.2多继承中的虚函数表

通过同样的方式打印,可以得到多继承中,派生类未重写的虚函数放在第一个继承的基类部分的虚函数表中。

5.3菱形继承、菱形虚拟继承

写在这里只是说明,这两种继承方式是存在的,由于太过复杂,这里就不再研究了。

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

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

相关文章

http协议报文头部结构解释

http协议报文头部结构 请求报文 报文解释 请求报文由三部分组成&#xff1a;开始行、首部行、实体主体 开始行&#xff1a;请求方法&#xff08;get、post&#xff09;url版本 CRLE 方法描述GET请求指定页面信息&#xff0c;并返回实体主体HEAD类似get要求&#xff0c;只不…

504 Gateway Time-out

问题描述 做Excel导入的功能&#xff0c;由于Excel的数据比较多&#xff0c;需要做处理然后入库&#xff0c;数据量大概200万&#xff0c;所以毫无悬念的导入Excel接口调用超过了一分钟&#xff0c;并且报错&#xff1a;504 gateway timeout。 解决方案 nginx超时限制。路径…

与WAF的“相爱相杀”的RASP

用什么来保护Web应用的安全&#xff1f; 猜想大部分安全从业者都会回答&#xff1a;“WAF&#xff08;Web Application Firewall,应用程序防火墙&#xff09;。”不过RASP&#xff08;Runtime Application Self-Protection&#xff0c;应用运行时自我保护&#xff09;横空出世…

微信小程序-----基础加强(二)

能够知道如何安装和配置vant-weapp 组件库能够知道如何使用MobX实现全局数据共享能够知道如何对小程序的API 进行 Promise 化能够知道如何实现自定义tabBar 的效果 一.使用 npm 包 小程序对 npm 的支持与限制 目前&#xff0c;小程序中已经支持使用 npm 安装第三方包&#x…

采用Java语言开发的(云HIS医院系统源码+1+N模式,支撑运营,管理,决策多位一体)

采用Java语言开发的&#xff08;云HIS医院系统源码1N模式&#xff0c;支撑运营&#xff0c;管理&#xff0c;决策多位一体&#xff09; 是不是网页形式【B/S架构]才是云计算服务? 这是典型的误区! 只要符合上述描述的互联网服务都是云计算服务&#xff0c;并没有规定是网页…

东软联合福建省大数据集团打造“数据要素×医疗健康”服务新模式

5月23日&#xff0c;东软集团与福建省大数据集团有限公司在福州签订战略合作协议。 据「TMT星球」了解&#xff0c;双方将在健康医疗数据要素价值领域展开合作&#xff0c;通过大数据服务&#xff0c;赋能商业保险公司的产品设计和保险两核&#xff0c;打造“数据要素医疗健康…

安卓分身大师4.6.0解锁会员安卓14可用机型伪装双开多开

需登录解锁会员功能&#xff0c;除了加速进入不能&#xff0c; 其他主要功能都是可以使用&#xff0c;由于验证较多一些功能需要特定操作使用&#xff0c;进行伪装时请不要直接伪装&#xff0c;先生成成功后再进行自定义伪装&#xff01;链接&#xff1a;https://pan.baidu.com…

机器人非线性控制方法——线性化与解耦

机器人非线性控制方法是针对具有非线性特性的机器人系统所设计的一系列控制策略。其中&#xff0c;精确线性化控制和反演控制是两种重要的方法。 1. 非线性反馈控制 该控制律采用非线性反馈控制的方法&#xff0c;将控制输入 u 分解为两个部分&#xff1a; α(x): 这是一个与…

vue 引入 emoji 表情包

vue 引入 emoji 表情包 一、安装二、组件内使用 一、安装 npm install --save emoji-mart-vue二、组件内使用 import { Picker } from "emoji-mart-vue"; //引入组件<picker :include"[people,Smileys]" :showSearch"false" :showPreview&q…

找钢集团亮相沙特利雅得建筑行业供应链展会

5月20日-21日&#xff0c;找钢产业互联集团&#xff08;以下简称&#xff1a;找钢集团&#xff09;亮相沙特利雅得建筑行业供应链展会。本次展会由沙特阿拉伯国家住房公司&#xff08;NHC&#xff09;主办&#xff0c;中信建设协办&#xff0c;涵盖住房新科技、绿色环保等多个主…

六零导航页 file.php 任意文件上传漏洞复现(CVE-2024-34982)

0x01 产品简介 LyLme Spage(六零导航页)是中国六零(LyLme)开源的一个导航页面。致力于简洁高效无广告的上网导航和搜索入口,支持后台添加链接、自定义搜索引擎,沉淀最具价值链接,全站无商业推广,简约而不简单。 0x02 漏洞概述 六零导航页 file.php接口处任意文件上传…

使用API有效率地管理Dynadot域名,进行域名邮箱的默认邮件转发设置

关于Dynadot Dynadot是通过ICANN认证的域名注册商&#xff0c;自2002年成立以来&#xff0c;服务于全球108个国家和地区的客户&#xff0c;为数以万计的客户提供简洁&#xff0c;优惠&#xff0c;安全的域名注册以及管理服务。 Dynadot平台操作教程索引&#xff08;包括域名邮…

DFS:解决二叉树问题

文章目录 了解DFS1.计算布尔二叉树的值思路代码展示 2.求根节点到叶节点数字之和思路代码展示 3.二叉树剪枝思路代码展示 4.验证二叉搜索树思路分析代码展示 5.二叉搜索树中第k小元素思路&#xff1a;代码展示 6.二叉树的所有路径思路分析代码展示 总结 了解DFS 所谓DFS就是就…

5.23.9 TransUNet:Transformers 为医学图像分割提供强大的编码器

TransUNet&#xff0c;它兼具 Transformers 和 U-Net 的优点&#xff0c;作为医学图像分割的强大替代方案。一方面&#xff0c;Transformer 对来自卷积神经网络 (CNN) 特征图的标记化图像块进行编码&#xff0c;作为用于提取全局上下文的输入序列。另一方面&#xff0c;解码器对…

Go知识点复习

Go知识点复习 1.关于包的使用和GOPATH的配置 src:用于以代码包的形式组织并保存Go源码文件, 需要手动创建pkg目录&#xff1a;用于存放经由go install命令构建安装后的代码包&#xff08;包含Go库源码文件&#xff09;的“.a”归档文件bin目录&#xff1a;与pkg目录类似&…

tomcat三级指导

版本 ./catalina.sh linux version.bat win 1.确认是否使用了tomcat管理后台 我们先找到配置文件&#xff1a;tomcat主目录下/conf/server.xml 可以查看到连接端口&#xff0c;默认为8080 然后查看manager-gui管理页面配置文件&#xff0c;是否设置了用户登录 配置文件…

如何创建 Gala Games 账户:解决 Cloudflare 验证指南 2024

Gala Games 站在数字娱乐新时代的前沿&#xff0c;将区块链技术与游戏相结合&#xff0c;重新定义了所有权和奖励。本文将引导您创建 Gala Games 账户并使用 CapSolver 解决 Cloudflare 验证难题&#xff0c;确保您顺利进入这一创新的生态系统。 什么是 Gala Games&#xff1f…

CRMEB开源商城标准版系统前端技术架构与实践探索

摘要&#xff1a; 随着电子商务的蓬勃发展&#xff0c;开源商城系统因其灵活性、可扩展性和成本效益受到了广泛关注。本文以CRMEB开源商城系统为例&#xff0c;探讨了其前端技术架构、开发实践及未来展望。通过对CRMEB系统前端技术的深入分析&#xff0c;旨在为开发者提供有价值…

vmware - 主机向虚拟机拷贝文件的临时方法

文章目录 vmware - 主机向虚拟机拷贝文件的临时方法概述笔记确认主机/虚拟机之间网络是通的在虚拟机中新建一个文件夹(e.g. c:\test), 将这个文件夹设为共享文件夹。查看虚拟机中的当前用户(远程登录要用)远程登录备注 - win8.1只能用mstscEND vmware - 主机向虚拟机拷贝文件的…

游戏行业 2024 Q1报告 | 国内同比上升7.6%,海外收入同比环比双增长,码住!

作为中国音像与数字出版协会主管的中国游戏产业研究院的战略合作伙伴&#xff0c;伽马数据发布了《2024年1—3月中国游戏产业季度报告》。 数据显示&#xff0c; 2024年1—3月&#xff0c;中国游戏市场实际销售收入726.38亿元&#xff0c;同比增长7.60%&#xff0c;主要受移动游…