0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

在C++中如何用虚函数实现多态

Android编程精选 来源:编程学习总站 作者:写代码的牛顿 2021-09-29 14:18 次阅读

01

C++虚函数探索

C++是一门面向对象语言,在C++里运行时多态是由虚函数和纯虚函数实现的,现在我们看下在C++中如何用虚函数实现多态。先来看一段代码。

// virtual_function.cpp : 此文件包含 “main” 函数。程序执行将在此处开始并结束。 // #include 《iostream》 class Base { public: Base()

{ std::cout 《《 “Base::constructor run” 《《 std::endl; } virtual void fun1()

{ std::cout 《《 “Base::fun1 run” 《《 std::endl; } virtual void fun2() { std::cout 《《 “Base::fun2 run” 《《 std::endl; } virtual ~Base()

{ std::cout 《《 “Base::desconstructor run” 《《 std::endl; } }; class Derive : public Base { public: Derive() { std::cout 《《 “Derive::constructor run” 《《 std::endl; } void fun1() { std::cout 《《 “Derive::fun1 run” 《《 std::endl; } void fun3()

{ std::cout 《《 “Derive::fun3 run” 《《 std::endl; } ~Derive() { std::cout 《《 “Derive::desconstructor run” 《《 std::endl; } }; int main() { Derive* d = new Derive(); d-》fun1(); d-》fun2(); d-》fun3(); delete d; }

这段代码编译运行后输出了:

Base::constructor run Derive::constructor run Derive::fun1 run Base::fun2 run Derive::fun3 run Derive::desconstructor run Base::desconstructor run

这段代码里基类Base定义了虚函数fun1和fun2,派生类Derive有成员函数fun1和fun3,其中派生类覆盖了继承而来的基类虚函数fun1。在主函数里创建Derive类型对象指针d指向Derive类型对象。由于派生类Derive成员函数fun1覆盖了基类Base成员函数fun1,因此通过d调用fun1实际调用的是派生类Derive类的成员函数fun1,而继承而来的成员函数fun2没有被覆盖,则通过指针d调用fun2实际调用的是基类成员函数fun2。这里好像让看不出虚函数有什么作用,那么我们将主函数修改如下:

int main() { Base* b = new Derive(); b-》fun1(); b-》fun2(); delete b; }

在程序里我们将创建一个基类指针b并指向的是派生类,并且调用delete释放内存时使用的是基类指针b。编译运行输出结果如下:

Base::constructor run Derive::constructor run Derive::fun1 run Base::fun2 run Derive::desconstructor run Base::desconstructor run

通过基类指针b调用fun1函数,实际调用的是派生类Derive的成员函数fun1,由于在派生类Derive中没有定义成员函数fun2,因此通过基类指针b调用fun2函数,实际调用的依旧是基类Base的成员函数fun2。代码里虽然我们没有对派生类的成员函数fun1加virtual,实际上派生类的成员函数fun1是虚函数。但是对于大多数C++初学者会有2个疑问的地方。1、通过基类指针b调用fun1函数,实际调用的是派生类的成员函数fun1。2、通过delete释放内存使用的是基类指针b,会调用派生类析构函数和基类析构函数,成功释放内存,不会存在内存泄露问题。

带有虚函数的类称为虚基类,子类继承虚基类。在C++中虚基类有一个虚函数表指针保存虚函数表地址,而虚函数表保存函数地址,虚函数表并不在虚基类里,但是虚函数表指针在虚基类里,子类继承虚基类,子类也就有了虚函数表指针。那么C++是如何通过虚函数表和虚函数表指针实现多态呢?打开VS2019,并用管理员身份运行“2019开发人员命令提示符”工具,如下图所示:

输入:cl /d1 reportSingleClassLayoutXXX [filename],XXX表示类名,[filename]表示类所在的.cpp文件路径。这里我输入源文件的派生类名和源文件路径,回车输出如下:

从输出可以看出派生类从基类继承了虚函数表指针vfptr,且占用字节数大小是4字节,刚好就是一个指针占用字节数。虚函数表vftable里保存了派生类成员函数fun1,基类成员函数fun2的地址,由于派生类成员函数fun3不是虚函数,因此虚函数表里没有fun3的地址。看到这里我们就明白了,通过基类指针b调用fun1的过程:通过虚函数表指针vfptr找到虚函数表vftable,再通过虚函数表找到派生类成员函数fun1的地址,调用派生类成员函数fun1。而通过基类指针b调用fun2的过程则是:通过虚函数表指针vfptr找到虚函数表vftable,再通过虚函数表找到基类成员函数fun2的地址,调用基类成员函数fun2。看到这里,第一个疑问已经解开了,关键在于虚函数表指针和虚函数表。

在C++中有虚函数的类,其析构函数默认是虚析构函数,只要是虚函数就会在虚函数表里有相应的函数地址,因此派生类里的虚函数表指针vfptr指向的虚函数表vftable必然保存着派生类析构函数的地址,类的析构过程:从继承链的最底端到最顶端依次调用析构函数,因此delete b调用过程:通过虚函数表指针vfptr找到虚函数表vftable,再通过虚函数表找到派生类析构函数地址,调用析构函数。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    3859

    浏览量

    61297
  • C++
    C++
    +关注

    关注

    21

    文章

    2066

    浏览量

    72880

原文标题:C++虚函数详解

文章出处:【微信号:AndroidPush,微信公众号:Android编程精选】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    什么是C++函数? 应该怎么定义? 用途是什么?

    什么是C++函数? 应该怎么定义? 主要用途是什么?
    发表于 11-08 06:58

    C++程序设计精简版

    章 指针第 7 章 自定义数据类型 第 3 篇 基于对象的程序设计第 8 章 类和对象第 9 章 关于类和对象的进一步讨论第 10 章 运算符重载 第 4 篇 面向对象的程序设计第 11 章 继承与派生第 12 章 多态性与函数
    发表于 10-09 07:26

    C++多态详解

    在编程语言和类型论中,多态(英语:polymorphism)指为不同数据类型的实体提供统一的接口。多态类型(英语:polymorphic type)可以将自身所支持的操作套用到其它类型的值上。
    发表于 09-20 17:18 322次阅读

    C++重载运算符和重载函数详解

    在计算机程序设计中,运算符重载(英语:operator overloading)是多态的一种。这里,运算符(比如+,=或==)被当作多态函数,它们的行为随着其参数类型的不同而不同。运算符并不一定总是符号。
    发表于 09-20 17:14 410次阅读

    C++生成Dll与回调函数测试

    描述了VS环境下,通过C++生成dll的方法,测试回调函数
    的头像 发表于 08-29 16:05 632次阅读
    <b class='flag-5'>C++</b>生成Dll与回调<b class='flag-5'>函数</b>测试

    如何为原生C++开发设置Android Studio

    。 为了进一步提高性能,您需要程序集使用程序集代码。 但是,使用ARM Neon内部函数可以避免编写汇编函数的复杂性。 相反,您只需要用C/C++编程并调用arm_neon.h头文件
    发表于 08-17 08:28

    C++演示的推理速度比Python演示的推理速度更快是为什么?

    同一主机机上采用相同型号的 Ran Object Detection C++ 演示 和 对象检测 Python 演示 。 C++ 演示的推理速度比 Python 演示
    发表于 08-15 06:52

    如何为Arm编译Cc++代码

    Studio或Keil MDK IDE环境。 Linux编译常见的编程语言Arm上得到了很好的支持,大多数开源工具都可以Linux发行版提供的包中使用。用于C++
    发表于 08-02 17:28

    C++ std::tie函数的作用和用法

    C++中std::tie函数的作用就是从元素引用中生成一个tuple元组,其在头文件中定义
    的头像 发表于 07-18 17:28 554次阅读

    嵌入式-C++函数的重载

    一、什么是函数的重载 两个以上的函数,具有相同的函数名,通过参数的类型和参数的个数不同。编译器自行匹配,自动确定调用哪一个函数 二、函数重载
    发表于 06-28 13:54

    何用C++实现一个线程池呢?

    C++线程池是一种多线程管理模型,把线程分成任务执行和线程调度两部分。
    发表于 06-08 14:53 781次阅读
    如<b class='flag-5'>何用</b><b class='flag-5'>C++</b><b class='flag-5'>实现</b>一个线程池呢?

    多态实现原理及其在面向对象编程中的应用

    在面向对象的编程中,多态性是一个非常重要的概念。
    的头像 发表于 06-08 14:19 403次阅读

    前端开发之函数式编程实践

    函数式编程是一种编程范式,它将程序抽象为函数和数据结构,通过函数调用来实现程序的功能,并且函数可以作为参数传递给其他
    发表于 06-05 10:19 334次阅读
    前端开发之<b class='flag-5'>函数</b>式编程实践

    S32和PowerPc artitechure是否支持项目的C++实现

    我想为 PowerPc MPC577C 实现 C++ 源代码。我更喜欢 freeGCC 作为编译器。我实施过程遇到了问题。我选择
    发表于 05-12 06:15

    c++常见函数

    c++常见函数集包括:线性代数方程组的解法、插值、数值积分、特殊函数函数逼近、随机数排序、特征值问题、数据拟合、方程求根和非线性方程组的解法、函数
    发表于 05-09 14:52 1次下载