C++学习之虚函数表与多态详解

 更新时间:2023年03月31日 15:09:12   作者:寡人正在Coding  
这篇文章主要为大家详细介绍了C++中虚函数表与多态的相关知识,文中的示例代码讲解详细,对我们学习C++有一定的帮助,感兴趣的小伙伴可以了解一下

概述

C++的多态在不同环境下实现方式可能不一样,虚函数表是C++实现多态的一种方式。

问题:

  • 什么情况下C++会使用虚指针和虚函数表?
  • 如果子类不新增任何虚函数,也不重写父类的虚方法,会和父类共用一张虚函数表么?
  • 父类的构造函数为什么不能正确的调用虚函数?

C++虚函数表指针和虚函数表

创建一个Base类

class Base
{
public:
	int a;
	int b;
};

查看Base内存布局

1>class Base size(8):
1> +---
1> 0 | a
1> 4 | b
1> +---

为Base类添加一个虚函数

class Base
{
public:
	int a;
	int b;

	virtual void BaseFunc1()
	{
		std::cout << "Call BaseFunc1 From Base" << std::endl;
	};
};

此时再查看Base类的内存布局

1>class Base size(12):
1> +---
1> 0 | {vfptr}
1> 4 | a
1> 8 | b
1> +---
1>Base::$vftable@:
1> | &Base_meta
1> | 0
1> 0 | &Base::BaseFunc1
1>Base::BaseFunc1 this adjustor: 0

Base类含有虚函数时,.rodata只读数据区会生成一个虚函数表,Base类会生成一个指向该虚函数表的指针成员变量。虚函数表存放.text代码区函数的地址。

再为Base添加一个虚函数

class Base
{
public:
	int a;
	int b;

	virtual void BaseFunc1()
	{
		std::cout << "Call BaseFunc1 From Base" << std::endl;
	};
	
	virtual void BaseFunc2()
	{
		std::cout << "Call BaseFunc2 From Base" << std::endl;
	}
};

查看Base类的内存分布

1>class Base size(12):
1> +---
1> 0 | {vfptr}
1> 4 | a
1> 8 | b
1> +---
1>Base::$vftable@:
1> | &Base_meta
1> | 0
1> 0 | &Base::BaseFunc1
1> 1 | &Base::BaseFunc2
1>Base::BaseFunc1 this adjustor: 0
1>Base::BaseFunc2 this adjustor: 0

Base类的虚函数表增加了一个新函数地址。

C++ 虚函数表和多态

为Base创建一个派生类Devire

class Derive : public Base
{
	//
};

查看Derive类的内存分布

1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Base::BaseFunc1
1> 1 | &Base::BaseFunc2

虚函数表的内容和父类Base一样

查看Base和Derive的虚函数表地址

Base和Derive并非公用一张虚函数表。

Derive重写父类Base的方法

class Derive : public Base
{
public:
	virtual void BaseFunc1() override
	{
		std::cout << "Call BaseFunc1 From Derive" << std::endl;
	}
};

查看Derive类的内存分布

1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Derive::BaseFunc1
1> 1 | &Base::BaseFunc2
1>Derive::BaseFunc1 this adjustor: 0

此时虚函数表的0元素被替换成了Derive::BaseFunc1的地址。

为Derive添加一个新的虚函数

class Derive : public Base
{
public:
	virtual void BaseFunc1() override
	{
		std::cout << "Call BaseFunc1 From Derive" << std::endl;
	}

	virtual void DeriveFunc1()
	{
		std::cout << "Call DeriveFunc1" << std::endl;
	}
};

再查看Derive类的内存分布

1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Derive::BaseFunc1
1> 1 | &Base::BaseFunc2
1> 2 | &Derive::DeriveFunc1
1>Derive::BaseFunc1 this adjustor: 0
1>Derive::DeriveFunc1 this adjustor: 0

Derive的虚函数表添加了一个新的函数地址。

让父类Base在构造函数中调用虚函数BaseFunc1。

class Base
{
public:
	Base()
	{
		BaseFunc1();
	}
	int a;
	int b;

	virtual void BaseFunc1()
	{
		std::cout << "Call BaseFunc1 From Base" << std::endl;
	};

	virtual void BaseFunc2()
	{
		std::cout << "Call BaseFunc2 From Base" << std::endl;
	}
};

class Derive : public Base
{
public:
	virtual void BaseFunc1() override
	{
		std::cout << "Call BaseFunc1 From Derive" << std::endl;
	}

	virtual void DeriveFunc1()
	{
		std::cout << "Call DeriveFunc1" << std::endl;
	}
};

int main()
{
	Derive d;
	return 0;
}

输出

Call BaseFunc1 From Base

虚函数的调用是错误的。

查看Derive和Base的构造函数的汇编代码

Base 构造函数汇编代码

...
00641F4D mov dword ptr [eax],offset Base::`vftable' (0649B34h)
{
BaseFunc1();
00641F53 mov ecx,dword ptr [this]
00641F56 call Base::BaseFunc1 (0641488h)
}
00641F5B mov eax,dword ptr [this]
...

Derive 构造函数的汇编代码

...
0064220A mov ecx,dword ptr [this]
0064220D call Base::Base (06412B2h)
{
00642212 mov eax,dword ptr [this]
00642215 mov dword ptr [eax],offset Derive::`vftable' (0649B40h)
//
}
...

观察汇编代码可知,构造Devire类的对象时,当调用父类Base的构造函数时,此时虚指针指向的虚函数表是父类Base的,只有调用Derive自己的构造函数时,虚指针被赋值为Derive的虚函数表,所以父类的构造函数不能正确的调用虚函数。

以上就是C++学习之虚函数表与多态详解的详细内容,更多关于C++虚函数表 多态的资料请关注脚本之家其它相关文章!

相关文章

  • 一篇文章带你了解C语言二分查找

    一篇文章带你了解C语言二分查找

    这篇文章主要为大家详细介绍了C语言二分查找法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • C语言函数超详细讲解上篇

    C语言函数超详细讲解上篇

    函数是一组一起执行一个任务的语句。每个 C 程序都至少有一个函数,即主函数 main() ,所有简单的程序都可以定义其他额外的函数,函数我们分两篇来讲解,接下来开始第一篇
    2022-04-04
  • string,CString,char*之间的转化

    string,CString,char*之间的转化

    下面是MFC/C++/C中字符类型CString, int, string, char*之间的转换的说明与举例,经常用的东西,相信对于用C/C++的朋友,还是比较有用的
    2013-03-03
  • C语言修炼之路一朝函数思习得 模块思维世间生上篇

    C语言修炼之路一朝函数思习得 模块思维世间生上篇

    函数是一组一起执行一个任务的语句。每个 C 程序都至少有一个函数,即主函数 main() ,所有简单的程序都可以定义其他额外的函数
    2022-03-03
  • C++ Qt开发之关联容器类使用方法详解

    C++ Qt开发之关联容器类使用方法详解

    当我们谈论编程中的数据结构时,顺序容器是不可忽视的一个重要概念,Qt 中提供了丰富的容器类,用于方便地管理和操作数据,本章我们将主要学习关联容器,主要包括 QMap ,QSet和 QHash,感兴趣的朋友跟着小编一起来学习吧
    2023-12-12
  • 基于QT制作一个简易的传输文件小工具

    基于QT制作一个简易的传输文件小工具

    本文主要介绍了通过QT实现的一个文件传输小工具。功能就是能实现文件的双向传输,即客户端能传给服务端,服务端可以传给客户端。文中示例代码具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2021-12-12
  • C++中设计一个类时的注意事项分享

    C++中设计一个类时的注意事项分享

    这篇文章主要来和大家分享一下C++中,设计一个类要注意哪些东西,这往往也是C++面试时会考到的问题,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-06-06
  • C++实现优酷土豆去视频广告的方法

    C++实现优酷土豆去视频广告的方法

    这篇文章主要介绍了C++实现优酷土豆去视频广告的方法,实例分析了C++实现屏蔽功能的相关技巧,需要的朋友可以参考下
    2015-04-04
  • exit和atexit的区别详细解析

    exit和atexit的区别详细解析

    以下是对exit与atexit的区别进行了详细的分析介绍,需要的朋友可以过来参考下
    2013-09-09
  • C语言实现程序开机自启动

    C语言实现程序开机自启动

    本文给大家分享的是一则C语言实现开机自启动的代码,主要是通过C来获取程序路径修改注册表项来实现,有需要的小伙伴可以参考下
    2016-01-01

最新评论