iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++学习之虚函数表与多态详解
  • 113
分享到

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

C++虚函数表 多态C++虚函数表C++ 多态 2023-05-14 05:05:08 113人浏览 泡泡鱼
摘要

目录概述c++虚函数表指针和虚函数表C++ 虚函数表和多态概述 C++的多态在不同环境下实现方式可能不一样,虚函数表是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++虚函数表 多态的资料请关注编程网其它相关文章!

--结束END--

本文标题: C++学习之虚函数表与多态详解

本文链接: https://www.lsjlt.com/news/202774.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++学习之虚函数表与多态详解
    目录概述C++虚函数表指针和虚函数表C++ 虚函数表和多态概述 C++的多态在不同环境下实现方式可能不一样,虚函数表是C++实现多态的一种方式。 问题: 什么情况下C++会使用虚指针...
    99+
    2023-05-14
    C++虚函数表 多态 C++虚函数表 C++ 多态
  • C++学习之多态的使用详解
    目录前言多态向上转型向下转型作用域前言 最近为了完成数据库系统的实验,又复习起了《C++ Primer》,上一次看这本巨著也是大二下的六月份,那时看面向对象程序编程这一章还云里雾里的...
    99+
    2024-04-02
  • C++ 虚拟函数详解:解答多态机制之谜
    c++++虚拟函数是实现多态性的关键,它允许派生类覆盖父类的行为:声明为虚函数,并在派生类中出现。调用时,编译器根据对象类型动态解析正确的函数版本。纯虚函数强制派生类提供自己的实现,以实...
    99+
    2024-04-29
    c++ 多态
  • c#基础学习之多态
    最近在看一本书《你必须知道的.Net》,书涵盖的内容比较多,对于c#,.Net平台以往所学的零散东西有了慢慢的总结和新的认识。把一些基础的东西记录于此。 先说说多态吧: 1.基类继承...
    99+
    2022-11-15
    c# 多态
  • C++多态特性之派生与虚函数与模板详细介绍
    目录继承与派生虚函数父类代码如下模板函数模板类模板字符串继承与派生 C ++ 是面向对象编程,那么只要面向对象,都会有多态、继承的特性。C++是如何实现继承的呢? 继承(Inheri...
    99+
    2024-04-02
  • C++的多态与虚函数你了解吗
    目录多态性虚函数总结多态性 多态性是面向对象程序设计的关键技术之一,若程序设计语言不支持多态性,不能称为面向对象的语言,利用多态性技术,可以调用同一个函数名的函数,实现完全不同的功能...
    99+
    2024-04-02
  • 深入了解C++的多态与虚函数
    目录1.多态的机制与虚函数的机制1.1 多态的机制1.2 虚函数的机制1.3虚函数表的结构图1.4 动态多态实现的三个前提件(很重要)2.多态实例应用3.多态的巨大问题与虚析构3.1...
    99+
    2024-04-02
  • C++ 虚函数及虚函数表详解
    多态”的关键在于通过基类指针或引用调用一个虚函数时,编译时不确定到底调用的是基类还是派生类的函数,运行时才确定。 #include <iostream> using ...
    99+
    2024-04-02
  • C++虚函数表与多态实例代码分析
    这篇文章主要介绍“C++虚函数表与多态实例代码分析”,在日常操作中,相信很多人在C++虚函数表与多态实例代码分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++虚函数表与多态实例代码分析”的疑惑有所帮助!...
    99+
    2023-07-05
  • 深入解析C++中的虚函数与多态
    1.C++中的虚函数C++中的虚函数的作用主要是实现了多态的机制。关于多态,简而言之就是用父类型别的指针指向其子类的实例,然后通过父类的指针调用实际子类的成员函数。这种技术可以让父类...
    99+
    2022-11-15
    虚函数 多态
  • C++学习之函数模板的使用详解
    目录C++函数模板重载的模板模板局限性显示具体化实例化和具体化模板函数类型的确定C++函数模板 template<typename T> void Swap(T &...
    99+
    2023-03-02
    C++函数模板使用 C++函数模板 C++函数
  • C++的多态与虚函数是什么
    这篇文章主要介绍“C++的多态与虚函数是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C++的多态与虚函数是什么”文章能帮助大家解决问题。多态性多态性是面向对象程序设计的关键技术之一,若程序设计...
    99+
    2023-06-29
  • C++学习之初始化列表详解
    目录前言一、类的初始化表二、initializer_list前言 本文主要介绍C++中地初始化列表 目前对初始化列表应该有两个方面的定义,一个是类的构造函数中使用的那个初始化表,另一...
    99+
    2023-03-19
    C++初始化列表 C++ 列表
  • C+继承之同名覆盖,函数重写与多态详解
    目录同名覆盖函数重写多态总结如果父类成员和子类成员名字相同是否允许?会发生什么? 同名覆盖 #include<iostream> using namespace st...
    99+
    2024-04-02
  • C++ 虚函数与纯虚函数代码详解
    目录什么是虚函数:虚函数的注意事项:存虚函数总结什么是虚函数: 虚函数 是在基类中使用关键字 virtual 声明的函数,在C++ 语言中虚函数可以继承,当一个成员函数被声明为虚函数...
    99+
    2024-04-02
  • python学习之Python count()函数详解
    大家好,依旧是你们的萧萧啊。 Python中的count()函数是一种非常常用的方法,用于统计某个元素在列表、元组或字符串中出现的次数。它的使用非常简单,只需要传入要统计的元素作为参数即可。在本篇博客中,我将详细介绍count()函数的用法...
    99+
    2023-09-29
    学习 python
  • Python学习之函数的定义与使用详解
    目录函数的定义函数的分类函数的创建方法-def函数的返回值-returnreturn与print的区别函数的传参必传参数默认参数不确定参数(可变参数)参数规则函数小练习函数的参数类型...
    99+
    2024-04-02
  • c++虚函数与虚函数表原理
    目录1.什么是虚函数2.虚函数会影响类的内存3.了解虚函数表--->通过虚函数表的指针去访问数据4.虚函数声明1.什么是虚函数 用virtual 修饰的成员函数叫虚函数 小知识...
    99+
    2024-04-02
  • C++虚函数和多态超详细分析
    目录1.什么是虚函数2.纯虚函数3.c++多态4.纯虚函数和ADT过程5.虚析构函数6.dynamic_cast类型转换7.成员函数指针 1.什么是虚函数 C++类中用virtual...
    99+
    2023-01-28
    C++虚函数和多态 C++虚函数 C++多态
  • C++中的多态问题—理解虚函数表及多态实现原理
    目录一、多态的概念概念构成条件二、虚函数的重写重写的定义重写的特殊情况override和final关键字区分重写、重载、重定义抽象类的概念三、多态的实现原理父类对象模型补充:生成默认...
    99+
    2023-02-05
    C++中的多态 C++虚函数表 C++多态实现原理
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作