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,一经查实,立即删除!

相关文章

go语言之变量

go是静态类型语言&#xff0c;因此变量是具有明确类型的&#xff0c;编译器也会检查变量类型的正确性 我们从计算机系统的角度来讲&#xff0c;变量就是一段或者多段内存&#xff0c;用于存储数据 文章目录 变量声明标准格式简便格式不指定变量类型批量声明简短格式 匿名变量变…

http协议报文头部结构解释

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

WXML模板语法-条件渲染和列表渲染

一、条件渲染 1.wx:if 在小程序中&#xff0c;使用wx:if"{{condition}}"来判断是否需要渲染该代码块,也可以用wx:elif和wx:else来添加else判断 // pages/list/list.js Page({data: {type:1} })<!--pages/list/list.wxml--><view wx:if"{{type 1}}&…

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): 这是一个与…

设计模式--观察者模式

观察者模式是一种行为设计模式&#xff0c;它定义了对象间的一种一对多的依赖关系&#xff0c;当一个对象的状态发生改变时&#xff0c;它的所有依赖者都会自动收到通知并更新。这种模式在许多应用场景中非常有用&#xff0c;例如在实现事件驱动编程、消息队列、发布-订阅模型以…

vue 引入 emoji 表情包

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

秒杀系统如何设计?【面试准备】

秒杀系统如何设计&#xff1f;【面试准备】 前言版权推荐秒杀系统如何设计&#xff1f;库存如何扣减的设计支付-延时队列最后 前言 2023-9-1 16:23:31 公开发布于 2024-5-22 00:09:02 以下内容源自《【面试准备】》 仅供学习交流使用 版权 禁止其他平台发布时删除以下此话…

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

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;包括域名邮…

头歌OpenGauss数据库-I.复杂查询第5关:至少学了某位学生(Oliver)所学的全部课程的学生

本关任务&#xff1a;根据提供的表和数据&#xff0c;查询至少学了Oliver同学所学的全部课程的其他同学的信息&#xff08;学号s_id&#xff0c;姓名s_name&#xff09;。 student表数据&#xff1a; s_ids_names_sex01Mia女02Riley男03Aria女04Lucas女05Oliver男06Caden男07Li…

DFS:解决二叉树问题

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

OceanBase V4.2 特性解析:MySQL兼容之rename column 语法支持

1. 背景描述 MySQL 自8.0版本起&#xff0c;已支持rename column语法&#xff0c;这一语法允许用户在不更改列定义的情况下&#xff0c;为列重新命名。下面为使用rename column语法的一个具体示例&#xff1a; alter table t1 rename column col_a to col_b; 在OceanBase过去…

如何秒杀Promise面试题

如何秒杀Promise面试题 如果你在面试的时候技术面给你出了点关于Promise的面试题首先不要慌&#xff0c;先问候他爹妈一套问候语&#xff01; 然后切记不要(ps:这是病句别在意!&#x1f923;) 自己想 找他要纸和笔 首先关于promise的面试题无非就是 promise 的状态和宏队列、…