C++ 多态以及多态的原理

文章目录

    • 多态的概念
    • 多态的构成条件
      • 虚函数的重写
        • 虚函数重写的两个例外
    • 重载、重写(覆盖)、重定义(隐藏)对比
    • C++11 final 和 override关键字
    • 抽象类
    • 接口继承和普通继承
    • 多态的原理
      • 虚函数表
      • 多态的原理
    • 单继承和多继承关系的虚函数表
      • 单继承中的虚函数表
      • 多继承中的虚函数表

多态的概念

多态的概念:通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会 产生出不同的状态
比如,买票时都是同一个景点有学生票半价和成人票全价等等

多态的构成条件

多态的构成条件主要涉及两个概念:虚函数和继承。
虚函数

  • 虚函数是C++中用于实现运行时多态性的关键概念。
  • 被virtual修饰的类成员函数称为虚函数
class Person {
public://虚函数virtual void BuyTicket() {//....}
};

继承中构成多态还需要满足两个条件

  • 被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写
  • 父类的指针或者引用进行调用

举个栗子

class Person {
public://虚函数virtual void BuyTicket() {cout << "买票-全价" << endl;}
};
class Student : public Person
{//重写基类函数virtual void BuyTicket(){cout << "买票-半价" << endl;}
};
//必须是基类对象指针或引用调用
void Func(Person& people)
{people.BuyTicket();
}
void Test()
{Person Mike;Func(Mike);Student Johnson;Func(Johnson);
}
int main()
{Test();return 0;
}

父类对象和子类对象调用同一个函数,得到的结果不一样

运行结果

image.png
解释

image.png

虚函数的重写

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

在上面的例子中 派生类Student中的BuyTicket函数就重写了基类Person的虚函数

虚函数重写的两个例外
  1. 协变
    派生类重写基类虚函数时,与基类虚函数返回值类型不同,且返回值必须是父子关系的指针或者引用。即基类虚函数返回基类对象的指 针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变
    比如
class Person
{
public:// 基类虚函数返回基类指针virtual Person* BuyTicket() {return new Person();}
};
class Student : public Person
{// 派生类协变,返回更具体的类型 Student*virtual Student* BuyTicket(){return new Student();}
};

上面例子中也构成虚函数的重写,派生类和基类的返回值不同,称为协变

  1. 析构函数的重写(基类与派生类析构函数的名字不同)
    如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字, 都与基类的析构函数构成重写,虽然基类与派生类析构函数名字不同。虽然函数名不相同, 看起来违背了重写的规则,其实不然,这里可以理解为编译器对析构函数的名称做了特殊处 理,编译后析构函数的名称统一处理成destructor。
class Person
{
public:virtual ~Person(){cout << "~Person()" << endl;}
};
class Student : public Person
{
public:~Student(){cout << "~Student()" << endl;}
};
int main()
{Person* p1 = new Person();Person* p2 = new Student();// 只有派生类Student的析构函数重写了Person的析构函数,下面的delete对象调用析构函数//才能构成多态,才能保证p1和p2指向的对象正确的调用析构函数。delete p1;delete p2;
}

运行结果:

image.png
12行加不加virtual关键字 都构成重写

重载、重写(覆盖)、重定义(隐藏)对比

image.png

C++11 final 和 override关键字

C++对函数重写的要求比较严格,有些情况可能由于疏忽,导致无法构成重写,这种情况编译器不会报错,程序会正常运行,但是得到的结果不是正确的,所以C++11引入了final和override关键字

  • final 修饰虚函数,表示该虚函数不能再被重写
class Person
{virtual void Func() final{cout << "virtual void Func() final" << endl;}
};class Student : public Person
{virtual void Func()//error{cout << "virtual void Func()" << endl;}
};

编译时报错:

image.png

  • override 检查派生类是否重写了基类虚函数。如果没重写,编译器会报错。
class Person
{
public:virtual void Func() const {cout << "virtual void Func() " << endl;}
};class Student : public Person
{
public:virtual void Func() override//error 派生类没有正确重写基类Func函数  编译器会报错,少了const修饰{cout << "virtual void Func()" << endl;}
};

抽象类

在虚函数的后面写上 =0 ,则这个函数为纯虚函数。包含纯虚函数的类叫做抽象类(也叫接口 类),抽象类不能实例化出对象。派生类继承后也不能实例化出对象,只有重写纯虚函数,派生类才能实例化出对象。纯虚函数规范了派生类必须重写,另外纯虚函数更体现出了接口继承。

class Person
{
public:virtual void Abstract() = 0;//纯虚函数  Person类为抽象类  Person类不能实例化出对象
};//派生类
class Student : public Person
{
public:virtual void Abstract() override//派生类必须重写纯虚函数,派生类才可以实例化出对象,否则不行{cout << "Hello World\n";}};
int main()
{//Person p1; //error 抽象类无法实例化对象Student s1;Person* s1prt = &s1;//使用基类指针访问s1prt->Abstract();return 0;
}

接口继承和普通继承

  • 普通继续 派生类继承了基类,可以使用函数,继承的是函数的实现
  • 接口继承,虚函数的继承是一种接口继承,派生类继承的是基类虚函数的接口,目的是为了重写,达成多态,继承的是接口。所以如果不实现多态,不要把函数定义成虚函数。

多态的原理

虚函数表

#include <iostream>
using namespace std;
class Person
{
public:virtual void Func();
private:int _a;char _b;
};
int main()
{cout << sizeof(Person) << endl;return 0;
}

上面代码求Person所占字节数大小
按照内存对齐的规则,Person类的大小应该是8(32位下)。
但实际结果是12

image.png
这里不仅要内存对齐,当实例化一个对象后发现,成员变量不仅仅只有_a,和_b 。还有一个指针_vfptr(虚函数表指针)

image.png

  • 一个含有虚函数的类都会有至少一个虚函数表指针,虚函数表指针存放在对象的前4个字节或者前8个字节(32位下4个字节,64位下8个字节)。
  • 而虚函数的地址会被存放到虚函数表中。虚函数表也简称为虚表
  • 虚函数表指针指向虚表。
  • 虚函数表本质是一个存虚函数指针的指针数组,一般情况这个数组最后面放了一个nullptr(VS编译器下做了处理,g++没有处理)

image.png
一个含有虚函数的类中都至少都有一个虚函数表指针,因为虚函数的地址要被放到虚函数表中,虚函数表也简称虚表。
通过下面的代码进行分析派生类中的虚表。

class Person
{
public:virtual void Func(){cout << "virtual void Func()" << endl;}virtual void Func2(){cout << "virtual void Func()" << endl;}void Func3()//普通函数{cout << "void Func3()" << endl;}
private:int _a = 0;char _b = 0;
};
class Student : public Person
{virtual void Func() override//重写基类函数{cout << "virtual void Func()" << endl;}
};
int main()
{	Person p1;//基类对象Student s1;//派生类对象return 0;
}
  • Person类中有两个虚函数,一个非虚函数。
  • Student继承了Person类,并且重写了Func函数。
  • 实例化出基类和派生类对象 监视窗口如下
    image.png
    可以发现:
  • 派生类对象也有一个虚表指针,虚表由两部分组成,一部分是继承基类的成员,另一部分是自己的成员。
  • 基类对象和派生类对象的虚表是不一样的,派生类重写了基类的Func虚函数,所以派生类对象虚表中存的是派生类重写后的函数地址。所以虚函数的重写也覆盖,覆盖值得是虚表中虚函数的覆盖,重写是语法的叫法,覆盖是底层的原理。
  • Func2虚函数被继承下来后也会被放到虚表中,Func3也会被继承下来,但是Func3不是虚函数,所以不会放到虚表中

多态的原理

class Person {
public://虚函数virtual void BuyTicket(){cout << "买票-全价" << endl;}
};
class Student : public Person
{//重写基类函数virtual void BuyTicket(){cout << "买票-半价" << endl;}
};
//必须是基类对象指针或引用调用
void Func(Person& people)
{people.BuyTicket();
}
void Test()
{Person p1;Func(p1);Student s1;Func(s1);
}

对于上面的例子

image.png

  • 当people指向的是基类对象时,people.BuyTicket();就会在基类对象p1中的虚表中找到对应虚函数
  • 当people指向的是派生类对象时,,people.BuyTicket();就会在派生类对象s1中的虚表中找到对应的虚函数
  • 通过虚表,实现了了不同对象去完成同一行为时,展现出不同的形态。

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

单继承中的虚函数表

class Base 
{
public:virtual void func1() { cout << "Base::func1" << endl; }virtual void func2() { cout << "Base::func2" << endl; }
private:int a;
};
class Derive :public Base 
{
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }virtual void func4() { cout << "Derive::func4" << endl; }
private:int _b;
};
int main()
{Base b1;Derive d1;return 0;
}
  • 上面代码中基类Base有两个虚函数
  • 派生类Derive继承了Base类,并且重写了func1函数,且新增了两个虚函数func3 和func4

单继承对象模型
image.png

image.png

通过监视窗口发现:派生类中新增的虚函数func3 和func4 没有进虚函数表。(不知道是编译器故意的 还是编译器的
bug)
image.png
我们可以利用程序自己打印虚表来观察,参考代码如下。

class Base 
{
public:virtual void func1() { cout << "Base::func1" << endl; }virtual void func2() { cout << "Base::func2" << endl; }
private:int a;
};
class Derive :public Base 
{
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }virtual void func4() { cout << "Derive::func4" << endl; }
private:int b;
};
typedef void (*VF_Ptr)();//函数指针
//VF_Prt table[];//函数指针数组//打印虚函数表
void PrintVFTable(VF_Ptr table[])
{for (int i = 0; table[i] != nullptr; ++i){printf("table[%d] = %p\n", i, table[i]);VF_Ptr Fun = table[i];//取出函数地址对其进行访问Fun();}cout << endl;
}
int main()
{Base b1;Derive d1;//虚函数表指针在对象的头四个字节(32位下), 拿到对象的地址对其强制类型转换:(int*)&p1//在解引用就能拿到对象前四个字节地址:*((int*)&p1),在将其强制类型转换位函数指针:(VF_Ptr*)(*(int*)&p1)PrintVFTable((VF_Ptr*)(*(int*)&b1));PrintVFTable((VF_Ptr*)(*(int*)&d1));return 0;
}

运行结果:

image.png
可以看出,不论是派生类还是基类,只要是虚函数都会存到虚表中

多继承中的虚函数表

class Base1 {
public:virtual void func1() { cout << "Base1::func1" << endl; }virtual void func2() { cout << "Base1::func2" << endl; }
private:int _a;
};
class Base2 {
public:virtual void func1() { cout << "Base2::func1" << endl; }virtual void func2() { cout << "Base2::func2" << endl; }
private:int _b;
};
class Derive : public Base1, public Base2 {
public:virtual void func1() { cout << "Derive::func1" << endl; }virtual void func3() { cout << "Derive::func3" << endl; }
private:int _d;
};

多继承对象模型
多继承对象模型对比单继承模型就复杂很多

image.png

image.png

image.png
派生类会有两个虚表,监视窗口仍然无法观察, 通过程序打印查看

typedef void(*VFPTR) ();
void PrintVTable(VFPTR vTable[])
{cout << " 虚表地址>" << vTable << endl;for (int i = 0; vTable[i] != nullptr; ++i){printf(" 第%d个虚函数地址 :0X%x,->", i, vTable[i]);VFPTR f = vTable[i];f();}cout << endl;
}
int main()
{Derive d;VFPTR* vTableb1 = (VFPTR*)(*(int*)&d);PrintVTable(vTableb1);//派生类第二个虚表指针需要加行Base对象大小的偏移量才能获得VFPTR* vTableb2 = (VFPTR*)(*(int*)((char*)&d + sizeof(Base1)));PrintVTable(vTableb2);return 0;
}

结果如下

image.png
和上面的对象模型一样。
可以发现:
多继承派生类的未重写的虚函数放在第一个继承基类部分的虚函数表中

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

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

相关文章

LJ3405-红外热释电处理芯片

描述&#xff1a; LJ3405 是一款专为热释电红外传感器信号放大及处理输出的数模混合专用芯片&#xff0c;内部集成了运算放大器、 双门限电压比较器、参考电压源、延时时间定时器和封 锁时间定时器及状态控制器等&#xff0c;专用于防盗报警系统、 人体门控制装置、照明控制开关…

Word2Vec的CBOW模型

Word2Vec中的CBOW&#xff08;Continuous Bag of Words&#xff09;模型是一种用于学习词向量的神经网络模型。CBOW的核心思想是根据上下文中的周围单词来预测目标单词。 例如&#xff0c;对于句子“The cat climbed up the tree”&#xff0c;如果窗口大小为5&#xff0c;那么…

[机缘参悟-122] :IT人如何认识自己的?自省、面试、考核、咨询?

目录 一、为什么要认识自己 二、认识自己的哪些方面&#xff1f; 三、如何认识自己 3.1 通过自省认识自己 3.2 通过面试认识自己 3.3 通过咨询认识自己 3.4 通过相亲认识自己 3.5 通过一段感情关系认识自己 一、为什么要认识自己 认识自己在人类的成长和心灵发展过程中…

光纤知识总结

1光纤概念&#xff1a; 光导纤维&#xff08;英语&#xff1a;Optical fiber&#xff09;&#xff0c;简称光纤&#xff0c;是一种由玻璃或塑料制成的纤维&#xff0c;利用光在这些纤维中以全内反射原理传输的光传导工具。 微细的光纤封装在塑料护套中&#xff0c;使得它能够…

CSND修改付费专栏价格

人工客服在个人中心右下角可以找到 客服回复已订阅专栏不支持修改价格

【ECShop电子商务系统__软件测试作业】ECSHOP系统搭建文档+接口测试用例+接口文档+接口测试脚本

一、选题题目可选《ECShop电子商务系统》、《EPShop电子商城系统》或者自选其它的开源系统(至少有十个以上的功能模块的系统&#xff0c;不得选功能少、简单的系统)。 软件测试作业 说明:接口测试相关资料 二、具体要求 1、搭建测试系统并写出搭建被测系统的全过程。 2、根…

Nginx介绍与安装

目录 nginx服务 1、Nginx 介绍 2、为什么选择 nginx 3、IO多路复用 1、I/O multiplexing【多并发】 2、一个请求到来了&#xff0c;nginx使用epoll接收请求的过程是怎样的? 3、异步&#xff0c;非阻塞 4、nginx 的内部技术架构 5、yum安装部署nginx和配置管理 1.获取…

Kafka集群部署 (KRaft模式集群)

KRaft 模式是 Kafka 在 3.0 版本中引入的新模式。KRaft 模式使用了 Raft 共识算法来管理 Kafka 集群元数据。Raft 算法是一种分布式共识算法&#xff0c;具有高可用性、可扩展性和安全性等优势。 在 KRaft 模式下&#xff0c;Kafka 集群中的每个 Broker 都具有和 Zookeeper 类…

Redis命令总结

1、启动Redis服务&#xff0c;登录Redis # 开启redis服务 redis-server redis配置文件路径例子&#xff1a; redis-server redis.windows.conf# 连接redis 【无密码】 redis-cli# 连接redis【有密码】 # 1 先连接再输入密码 redis-cli auth 密码 2、连接时输入 IP址、端口号、…

仿蓝奏云网盘 /file/list SQL注入漏洞复现

0x01 产品简介 仿蓝奏网盘是一种类似于百度网盘的文件存储和共享解决方案。它为用户提供了一个便捷的平台,可以上传、存储和分享各种类型的文件,方便用户在不同设备之间进行文件传输和访问。 0x02 漏洞概述 仿蓝奏云网盘 /file/list接口处存在SQL注入漏洞,登录后台的攻击…

烟火检测/区域人流统计/AI智能分析网关V4如何配置通道?

TSINGSEE青犀智能分析网关&#xff08;V4版&#xff09;是一款高性能、低功耗的软硬一体AI边缘计算硬件设备&#xff0c;硬件内部署了近40种AI算法模型&#xff0c;支持对接入的视频图像进行人、车、物、行为等实时检测分析&#xff0c;并上报识别结果&#xff0c;并能进行语音…

JS 监听网络状态

我们在开发过程中会遇到监听用户网络状态的需求&#xff0c;通过JS可以获取当前的网络状态&#xff0c;包括下载速度、网络延迟、网络在线状态、网络类型等信息 具体获取如下&#xff1a; let info navigator.connection console.log(info)可以看到&#xff0c;包含几个信息…