C++三大特性—多态 “抽象类与虚函数表”

news2024/11/23 6:40:37

抽象类和虚函数表是 C++中实现多态性的重要概念,它们对于学习 C++非常重要。
掌握抽象类和虚函数表的使用方法对于理解 C++的多态性是非常重要的。在 C++中,通过使用抽象类和虚函数表,可以实现基于多态性的各种功能,如继承、多态、模板等。同时,在实际应用中,抽象类和虚函数表也是常用的设计模式之一,如抽象工厂模式、观察者模式等。

目录

  • 抽象类
    • 接口继承和实现继承
  • 虚函数表
  • 多继承关系的虚函数表

抽象类

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

 和普通的虚函数不一样,一个纯虚函数无需定义。而且 =0 只能出现在类内部的虚函数声明处。我们也可以为虚函数提供定义,不过函数体必须定义在类的外部。也就是说,我们不能在类的内部为一个纯虚函数提供函数体。

class Person
{
public:
	virtual void work() = 0;//纯虚函数
};
class Student : public Person
{
public:
	virtual void work()
	{
		cout << "Student-学生上学" << endl;
	}
};
class Teacher : public Person
{
public:
	virtual void work()
	{
		cout << "Teacher-老师教书" << endl;
	}
};
	Person obj;       //错误写法,Person类中声明了纯虚函数,不能定义Person的对象
	Student obj_stu;  //正确写法
	Teacher obj_per;  //正确写法

 派生类继承基类的纯虚函数,如果不重写虚函数,那么这个派生类仍然是抽象类。所以必须对其进行重新定义(重写)才能实例出对象:
在这里插入图片描述

接口继承和实现继承

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


虚函数表

在了解虚函数表之前我们先来看一道常见的面试题:

//sizeof(Base)是多少?
class Base
{
public:
	virtual void Func1()
	{
		cout << "Func1()" << endl;
	}
private:
	int _b = 1;
	char _ch;
};

我们很可能会简单的以为只是常见的类的内存对齐问题,认为结果是8,但是结果是:
在这里插入图片描述
为什么结果是12呢?原因就是有虚函数表的存在。
在这里插入图片描述
在这里插入图片描述

注意:我们要理清楚虚函数表(虚表)与虚基表的区别,不要搞混(虚基表)


那么派生类重写(覆盖)基类的虚函数,虚函数表又是以什么样的形式变化呢?

class Base
{
public:
	virtual void Func1()
	{
		cout << "Base::Func1()" << endl;
	}
	virtual void Func2()
	{
		cout << "Base::Func2()" << endl;
	}
	void Func3()
	{
		cout << "Base::Func3()" << endl;
	}
private:
	int _b = 1;
	char _ch;
};
class Derive : public Base
{
public:
	virtual void Func1()
	{
		cout << "Derive::Func1()" << endl;
	}
private:
	int _d = 2;
};

在这里插入图片描述

完成重写的虚函数虚表对应的位置覆盖成重写的虚函数。、


那么虚函数表是存放在哪个区的呢?栈?堆?静态区?常量区?
我们通过下面代码可以大致猜测一下:

int main()
{
	int a = 0;
	cout << "栈:" << &a << endl;

	int* p1 = new int;
	cout << "堆:" << p1 << endl;

	const char* str = "hello world";
	cout << "代码段/常量区:" << (void*)str << endl;

	static int b = 0;
	cout << "静态区/数据段:" << &b << endl;

	Base be;
	cout << "虚表:" << (void*)*((int*)&be) << endl;
	return 0;
}

在这里插入图片描述
运行结果:在这里插入图片描述
所以我们猜测虚表应该存放在静态区
这里给大家提供一个打印虚表的办法:

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()
{
	Base b;
	Derive d;
	VFPTR * vTableb = (VFPTR*)(*(int*)&b);
	PrintVTable(vTableb);
	VFPTR* vTabled = (VFPTR*)(*(int*)&d);
	PrintVTable(vTabled);
	return 0;
}

思路:取出b、d对象的头4bytes,就是虚表的指针,前面我们说了虚函数表本质是一个存虚函数指针的指针数组,这个数组最后面放了一个nullptr
1.先取b的地址,强转成一个 int * 的指针
2.再解引用取值,就取到了b对象头4bytes的值,这个值就是指向虚表的指针
3.再强转成 VFPTR *,因为虚表就是一个存VFPTR类型(自己重定义的虚函数指针类型)的数组。
4.虚表指针传递给PrintVTable进行打印虚表
5.需要说明的是这个打印虚表的代码经常会崩溃,因为编译器有时对虚表的处理不干净,虚表最后面没有放nullptr,导致越界,这是编译器的问题。我们只需要点目录栏的 - 生成 - 清理解决方案,再编译就好了。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
相同类型的对象共用一个虚表


多继承关系的虚函数表

且看下面代码分析:

class Base1 {
public:
	virtual void func1() { cout << "Base1::func1" << endl; }
	virtual void func2() { cout << "Base1::func2" << endl; }
private:
	int b1;
};
class Base2 {
public:
	virtual void func1() { cout << "Base2::func1" << endl; }
	virtual void func2() { cout << "Base2::func2" << endl; }
private:
	int b2;
};
class Derive : public Base1, public Base2 {
public:
	virtual void func1() { cout << "Derive::func1" << endl; }
	virtual void func3() { cout << "Derive::func3" << endl; }
private:
	int d1;
};

主要的关系就是Derive同时继承Base1与Base2 ,Derive重写了func1但没有重写func2,且自生还有一个func为虚函数。

多继承虚函数表:
在这里插入图片描述
我们使用上面说过的打印虚表的方法:

	Derive d;
	VFPTR* vTableb1 = (VFPTR*)(*(int*)&d);
	PrintVTable(vTableb1);
	VFPTR* vTableb2 = (VFPTR*)(*(int*)((char*)&d + sizeof(Base1)));
	PrintVTable(vTableb2);

在这里插入图片描述
在这里插入图片描述


如有错误或者不清楚的地方欢迎私信或者评论指出🚀🚀

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

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

相关文章

极简JVM结构图示

参考资料 JVM极简教程 JVM结构 JIT编译器&#xff0c;对于经常需要执行的字节码进行 类加载子系统 类加载器 tomcat的自定义类加载器 为了进行类的隔离&#xff0c;如果Tomcat直接使用AppClassLoader类加载类&#xff0c;那就会出现如下情况&#xff1a; 应用A中有个com.e…

Android中Binder在项目中的具体使用详解

前些天发现了一个蛮有意思的人工智能学习网站,8个字形容一下"通俗易懂&#xff0c;风趣幽默"&#xff0c;感觉非常有意思,忍不住分享一下给大家。 &#x1f449;点击跳转到教程 前言&#xff1a;Binder的介绍 在 Android 中&#xff0c;Binder 是一种跨进程通信&…

静默安装oracle

oracle依赖环境包 一、创建用户属组 [rootlocalhost ~]# groupadd oinstall[rootlocalhost ~]# groupadd dba[rootlocalhost ~]# groupadd oper[rootlocalhost ~]# useradd -g oinstall -G dba,oper oracle[rootlocalhost ~]# passwd oracle #修改oracle用户密码 二、创建目录…

二叉搜索树(BST)详解及代码实现

推荐可视化插入、删除节点的二叉树网站&#xff1a;Binary Search Tree Visualization (usfca.edu) 1. 概述 二叉搜索树&#xff08;Binary Search Tree&#xff0c;简称BST&#xff09;是一种特殊的二叉树结构&#xff0c;它具有以下特点&#xff1a; 有序性&#xff1a;对于…

Eureka入门 ; 服务注册中心,服务注册服务发现;SpringCloud eureka

一、引入 Spring Cloud封装了netflix公司的Eureka模块来进行实现服务治理。 在传统的RPC远程调用框架中&#xff0c;管理每个服务服务之间依赖关系比较复杂&#xff0c;所以需要服务治理&#xff0c;管理服务之间的依赖。可以实现服务注册、调用、负载均衡、容错等技术。 1. 服…

2023九坤投资暑期实习笔试复盘

5.22号笔试&#xff0c;5.24确认自己笔试挂。想想这也是自己第一次做量化私募基金的笔试&#xff0c;在此复盘一下。情况&#xff1a;北邮本硕。但开始准备暑期准备的比较晚&#xff0c;4月初才开始一边刷题一边投简历&#xff0c;所以手撕算法不太强&#xff0c;但运气和灵感好…

ChatGPT除了模型, 各个大厂、中厂、小厂们还在卷什么?

ChatGPT 问世后&#xff0c;各大公司都在「大模型」上下了苦功&#xff0c;模型能力也成为大家最关注的话题。ChatGPT 虽直接定义了基于大模型的生成式对话机器人这个产品类型&#xff0c;其模型的强大也导致大家都忽略了它在产品使用上的问题。 其实不管是底层模型开发的大厂…

印尼市场入门指南:品牌如何在当地获得市场份额?

2023年&#xff0c;印尼成为了全球最大的新兴市场之一。印尼是东南亚最大的经济体&#xff0c;拥有庞大的人口和潜在的消费市场&#xff0c;吸引着越来越多的国际品牌进入。根据预测&#xff0c;印尼的消费支出将在2023年达到1.3万亿美元&#xff0c;成为亚洲增长最快的消费市场…

个人博客搭建详细步骤

1. 安装 jdk 和 tomcat 下面将带大家安装 jdk 和部署 tomcat; 首先在本地下载好 jdk 和 tomcat 安装压缩包在服务器新建一个目录&#xff0c;比如在服务器新建一个目录 soft&#xff0c;上传 jdk, tomcat 到服务器 mkdir soft cd soft rz 选择上传的文件名称 //上传文件新建…

【windows脚本】使用diskpart命令管理未分配磁盘

环境 系统&#xff1a;win10 x64 概述 使用windows脚本管理未分配磁盘&#xff0c;手动操作需要做以下几步&#xff1a; 1、初始化磁盘GPT形式&#xff1b; 2、新建简单卷&#xff0c;设置大小和驱动器号。 3、格式化。 diskpart命令 使用diskpart工具&#xff0c;命令如…

PNAS| 绘制人脑发育曲线:横断面研究低估了人脑变化

文章目录 第一部分&#xff1a;横断面数据低估了与年龄相关的大脑变化。第二部分&#xff1a;横截面规范模型最小限度地帮助个体化预测。第三部分&#xff1a;非年龄相关因素对于预测个体变化的误差起到了贡献作用。参考 **按&#xff1a;**研究人脑毕生发展对于理解常见的精神…

springboot应用程序并发请求达到多少会被拒绝

文章目录 前言内嵌服务器tomcat测试 undertow 前言 前几天刷B站看到了一个问题&#xff0c;感觉挺有意思的&#xff0c;“忽略接口本身性能&#xff0c;springboot应用程序能处理多少请求不被拒绝&#xff1f;”&#xff0c;今天便来探一探这个问题 本文主要重点是侧重于spri…

matplotlib常用函数总结

文章目录 1.IDE里的1.1 显示模式&#xff08;plt.ion()和plt.ioff()&#xff09;1.2 backend说明 2 jupyter里的3 通用的3.1 cmap3.2 subplot()相关3.3 绘制动态图&#xff08;Animation类&#xff09;3.4 matplotlib利用rcParams配置样式参数 4. 与opencv连用可能遭遇的问题1.…

迭代器的设计原则

iterator必须提供5种associated types 1.iterator_category 2.value_type 3.diiference_type 4.pointer 5.reference 对于iterator_category来说&#xff0c;例如双向链表:typedef bidirectional_iterator_tag iterator_category;算法和迭代器的关系&#xff1a; 算法提问&a…

WAV 格式和音频裁剪、转码处理

文章目录 0、参考资料1、WAV 格式了解1.1 WAV 文件头1.2 RIFF Chunk 区块1.3 Format Chunk 区块1.4 Data Chunk 区块 2、音频剪裁 -> 解码 -> 编码2.1 mp32.1.1 裁剪2.1.2 解码2.1.3 编码 2.2 pcm 裁剪 0、参考资料 【音频处理】WAV 文件格式分析 ( 逐个字节解析文件头 …

用友BIP新一代全球司库,重塑企业资金管理新价值

资金是企业有效推进业务运营与发展并赖以生存的血液和养料&#xff0c;资金管理是企业财务管理的核心职能&#xff0c;在《“十四五”中央企业发展规划纲要》中明确央企要将集团资金管理业务规划置于重要战略位置。而国资委1号文发布&#xff0c;则进一步强调司库建设对中央企业…

2核2G3M腾讯云轻量应用服务器CPU性能测评

阿里云轻量应用服务器2核2G3M带宽一年95元&#xff0c;100%CPU性能&#xff0c;3M带宽下载速度384KB/秒&#xff0c;40GB SSD系统盘&#xff0c;月流量200GB&#xff0c;折合每天6.6GB流量&#xff0c;超出月流量包的流量按照0.8元每GB的支付流量费&#xff0c;地域节点可选广州…

vue+springboot前后端分离项目整合部署流程

我们在进行前后端分离开发的时候&#xff0c;一般是将前端项目部署到nginx服务器上&#xff0c;与后端项目分开部署&#xff0c;但是如果是个人开发的小型项目&#xff0c;不需要这么麻烦&#xff0c;直接将前后端项目放到一起部署即可。 本文就来介绍一下前后端项目一起部署的…

第一章 部署DHCP服务

♥️作者介绍&#xff1a;奇妙的大歪 ♥️个人名言&#xff1a;但行前路&#xff0c;不负韶华&#xff01; ♥️个人简介&#xff1a;云计算网络运维专业人员 目录 DHCP&#xff1a;动态主机配置协议 1.DHCP的优点 2.DHCP的分配方式 3.DHCP的工作原理&#xff08;租约过程…

winows搭建远程仓库Github(linux通用)

winows搭建远程仓库Github&#xff08;linux通用&#xff09; 文章目录 winows搭建远程仓库Github&#xff08;linux通用&#xff09;创建远程仓库配置SSH克隆项目多人协同开发代码冲突标签分支 创建远程仓库 以下操作为演示在Github网站上创建远程仓库 1.登陆注册Github 2.创…