我的编程空间,编程开发者的网络收藏夹
学习永远不晚

波奇学C++:多态知识点

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

波奇学C++:多态知识点

多态中函数的重写(基类指针访问派生类函数),只重写函数的实现,而不重写声明。

class Person{public: virtual void fun(int i = 0){cout << "Person"<<" "<fun();return 0;}

结果是 student 0 原因在于重写时只重写函数的实现,就是说相当于Person的fun的声明和Student的函数实现的拼在一起所以缺省值是0。

为什么多态调用(重写)只能用父类的指针和引用,不能子类指针或者引用,不能是父类对象?

如果是子类指针或者引用就不是多态调用了只是单存子类对父类的重定义,隐藏函数。

上一篇文章提到的,多态的本质就是基类和派生类的虚表中保存的虚函数地址被覆盖了,多态调用意味着访问的必须是子类的虚表而不是父类的。

子类对象直接赋值父类不会拷贝虚表虚函数的地址

上图为赋值前,下图为赋值后,如图他们的__vfptr始终不同,所以父类对象必然无法访问子类对象的虚函数的地址。

为什么指针和引用可以?

父类型指针表示它范围的范围是父类,所以它指向子对象时,本质上依然说访问子类的父类部分,虚表依然是子类的虚表。

引用同理,相当于切割出子类中父类的部分。本质上依然是子类的虚表。

为什么父类指针可以指向子类对象?可以指向意味着结构相似,原因在于,继承相当于把父对象一整个拷贝放在子对象中,结构相似也是向上转换的基础

虚表的存储在代码段

证明思路:输出各个区的地址和虚表的地址,进行比较,字节相差较少说明在哪个区。

int main(){Person p;Student st;int a = 1;printf("栈上:%x\n", &a);int* b = new int;printf("堆上:%x\n", b);static int c = 0;printf("静态区:%x\n", &c);const char* d = "abcde";printf("代码段:%x\n", d);printf("虚表1:%x\n", *((int*)&p));printf("虚表2:%x\n", *((int*)&st));return 0;}

注意打印对象是打印对象的成员变量的值,这里是因为__vptr内置变量(保存虚表地址)的在成员变量首位,所以可以打印出来,同时int* 是只取虚表的地址后四个字节(小端机),%x也是只打印地址的后4位字节。

通过比较可发现,虚表和代码段的位置更近,所以虚表在代码段中。

派生类新的虚函数保存在虚表中,原有虚函数的地址的下面

class Person{public: virtual void fun(int i = 0){cout << "Person"<<" "<

fun1()是Student的虚函数,fun1保存在子函数的虚表中

证明:虚表保存函数指针地址,虚表可以看成指针数组,所以我们可以把虚表的函数指针打印出来。

typedef void(*FUNC_PTR) ();//重定义函数指针类型//形参是数组,实参为数组指针void PrintVFT(FUNC_PTR table[]){    //vs会在虚表末尾保存一个空指针,所以循环到nullptr为止for (size_t i = 0; table[i] != nullptr; i++){printf("[%d]:%p\n", i, table[i]);}}int main(){Person ps;Student st;int vft1 = *((int*)&ps);//86位机器地址是32位转换成int*PrintVFT((FUNC_PTR*)vft1);int vft2 = *((int*)&st);PrintVFT((FUNC_PTR*)vft2);return 0;}

 如图上面为父类虚表保存的地址,下面为派生类虚表保存的指针地址。重写的虚函数覆盖了原有的地址,并且新地址在虚表内。

静态多态:指的是函数重载,指的是编译的时候函数地址确定了

动态多态:继承,虚函数重写,调用的函数地址的确定是在运行时去虚表中确定的

多继承的多态问题

typedef void(*FUNC_PTR) ();void PrintVFT(FUNC_PTR table[]){for (size_t i = 0; table[i] != nullptr; i++){printf("[%d]:%p", i, table[i]);FUNC_PTR f = table[i];f();printf("\n");}}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;};int main(){Derive d;cout << sizeof(d) << endl;int vft1 = *((int*)&d);Base2* ptr = &d;int vft2 = *((int*)ptr);PrintVFT((FUNC_PTR*)vft1);PrintVFT((FUNC_PTR*)vft2);return 0;}

下面代码的Derive继承了Base1和Base2,其中两个fun1()都被继承了。

打印结果

 为什么是20?

 因为是一整个对象继承,所以会存在两个虚表,base1,base2虚表指针+int变量 8+8+4=20

由上面的结果图可知fun1在两个虚表中被重写,且都调用了同一个函数。但是地址却不一样,

实际上调用虚表2的fun()的地址,会改变指针位置和虚表1fun()指针相同,再调用函数。

反汇编证明

b1,b2指针分别调用fun1(),反汇编,call指令进入func1函数,此时

注意此处fun1()的地址是0C92840h

调用base2的fun1虚表地址,此时地址是0C94670h

进入call指令,ecx-8,再jump向0C91244h地址最后到base1虚表的地址。

 

 简单来说指向base2部分的指针,先指向base1的,再调用指针1保存的重写函数的地址。

来源地址:https://blog.csdn.net/Bagayalu1234567/article/details/132762743

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

波奇学C++:多态知识点

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

下载Word文档

猜你喜欢

Java多态知识点有哪些

本篇内容主要讲解“Java多态知识点有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java多态知识点有哪些”吧!1、将某个对象的引用视为其基类对象的引用的做法被称作“向上转型”(upcas
2023-06-03

C#多线程的知识点有哪些

这篇文章主要介绍“C#多线程的知识点有哪些”,在日常操作中,相信很多人在C#多线程的知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C#多线程的知识点有哪些”的疑惑有所帮助!接下来,请跟着小编一起来
2023-06-17

JAVA 面向对象之多态的知识点有哪些

本篇内容介绍了“JAVA 面向对象之多态的知识点有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 概念多态是面向对象程序设计(OOP
2023-07-02

java继承多态和抽象类接口知识点总结

本篇内容主要讲解“java继承多态和抽象类接口知识点总结”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“java继承多态和抽象类接口知识点总结”吧!一、继承通过扩展一个已有的类,并继承该类的属性和
2023-06-03

Java多线程的其他知识_动力节点Java学院整理

一、线程组 /** * A thread group represents a set of threads. In addition, a thread * group can also include other thread gr
2023-05-31

android初学者必须掌握的Activity状态的四大知识点(必读)

这几天一直都在捣鼓android的知识点,兴趣班的老师,讲课太过深奥,天(想到什么就见什么,后后面完全不想听),最后自己找资料总结了在Android学习中很重要的一个组件Activity,那就开始吧! 第一:掌握Activity的四种状态及
2022-06-06

PHP继承与多态备考指南:彻底掌握知识点,顺利通过考试

PHP继承与多态是面向对象编程(OOP)中的重要概念,也是PHP考试的必考知识点。掌握继承和多态的知识,对于理解和使用OOP有很大的帮助。本文将详细介绍PHP继承与多态的概念、特性和用法,并提供示例代码帮助理解。
PHP继承与多态备考指南:彻底掌握知识点,顺利通过考试
2024-02-04

c#多线程—基础概念到“双色球”项目实现(附知识点目录、代码、视频)

总结:视频中对于多线程讲的非常透彻,从线程基础概念—>.net不同版本出现的线程方法—>多线程常出现问题—>双色球项目实践,每个知识点都有代码实操,受益匪浅。附上学习笔记和实操代码。 视频 目录 一、线程、进程概念及优缺点二、四种异
2023-08-30

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录