引言

C++相对其他面向对象语言来说,之所以灵活、高效。很大程度的占比在于其多态技术和模板技术。C++虚函数表是支撑C++多态的重要技术,它是C++动态绑定技术的核心。

如果对多态还不了解的小伙伴,可以点这里C++多态详解基础篇。

在不考虑继承的情况下,如果一个类中有虚函数,那么这个类就有一个虚函数表,这个虚函数表在编译期间确定,这个类对象共享。而这个类所有的实例化对象中都有一个虚函数指针,这个虚函数指针就指向同一份虚函数表。

一、多态的使用及内存分布图

假设现在有个基类的class A,A中有虚函数,class B继承了A,并且B重写了A中的虚函数。那么,我们在使用多态的时候,通常会有两种方式:

// 方式一
class A* a = new class B;
// 方式二
class B b;
class A *a = &b;

上面两种方式,都是用父类的指针指向了子类的对象。区别在于,方式一的子类对象是new出来的,存在于堆区;方式二的子类对象则保存在栈区。

二、多重继承的虚函数表

2.1 代码示例

class A {public:void func1(){ std::cout << "Class A func1" << std::endl; }void func2(){ std::cout << "Class A func2" << std::endl; }virtual void v_func1(){ std::cout << "Class A v_func1" << std::endl; }virtual void v_func2(){ std::cout << "Class A v_func2" << std::endl; }private:int m_aMember;
};class B : public A{public:void func1(){ std::cout << "Class B func1" << std::endl; }virtual void v_func1(){ std::cout << "Class B v_func1" << std::endl; }private:int m_bMember;
};class C : public B{public:void func2(){ std::cout << "Class C func2" << std::endl; }virtual void v_func2(){ std::cout << "Class C v_func2" << std::endl; }private:int m_bMember;
};int main(int argc, char* argv[])
{std::cout << "============== class B : public A ============" << std::endl;// class B  b;// class A *a = &b;class A* a = new class B;a->func1();a->func2();a->v_func1();a->v_func2();return 0;
}

上面的代码中:

  • 父类A中,有属性m_aMember,普通函数func1() func2(),有虚函数 v_func1() v_func2()
  • class B继承了A,有属性m_bMember,普通函数func1(),虚函数 v_func1()
  • class C继承了B,有属性m_cMember,普通函数func2(),虚函数 v_func2()

注意,父类的虚函数表和子类的虚函数表不是同一张表,虚函数表是在编译时确定的,虚函数保存在代码段,仅有一份,属于类而不属于某个具体的实例对象

2.2 多重继承图解

B继承于A,B的虚函数表示在A的虚函数表的基础上有所改动。改的部分就是子类B重写的父类A的虚函数v_func1()。假设此时B没有重写A任何一个虚函数,那么B类的虚函数表和类A的虚函数表的内容是相同的。

运行结果分析:

毫无疑问,对于A类型的指针来说,可见的部分只有图中红色框的部分。因为 B重写了A的虚函数v_func1(),所以在B的虚函数表会覆盖父类A的虚函数v_func1()。所以,会调用到B的虚函数v_func1(),从而实现多态。

同理,不难猜出C的逻辑结构图:C继承于B,B继承于A

int main(int argc, char* argv[])
{std::cout << "C -> B ->  A  " << std::endl;std::cout << "======= class A* a = new class C =======" << std::endl;class A* ac = new class C;ac->func1();ac->func2();ac->v_func1();ac->v_func2();std::cout << "======= class B* b = new class C =======" << std::endl;class B* bc = new class C;bc->func1();bc->func2();bc->v_func1();bc->v_func2();return 0;
}

运行结果:

三、多继承的虚函数表

多继承指的是一个类同时继承多个基类,如果每个基类都有虚函数,那么对应的每个基类都有自己的虚函数表。

class A {public:void func1(){ std::cout << "Class A func1" << std::endl; }void func2(){ std::cout << "Class A func2" << std::endl; }virtual void v_func1(){ std::cout << "Class A v_func1" << std::endl; }virtual void v_func2(){ std::cout << "Class A v_func2" << std::endl; }private:int m_aMember;
};class B {public:void func1(){ std::cout << "Class B func1" << std::endl; }virtual void v_func1(){ std::cout << "Class B v_func1" << std::endl; }virtual void v_func2(){ std::cout << "Class B v_func2" << std::endl; }virtual void v_func4(){ std::cout << "Class B v_func4" << std::endl; }private:int m_bMember;
};class C : public A, public B{public:void func1(){ std::cout << "Class C func1" << std::endl; }virtual void v_func1(){ std::cout << "Class C v_func1" << std::endl; }virtual void v_func2(){ std::cout << "Class C v_func2" << std::endl; }virtual void v_func3(){ std::cout << "Class C v_func3" << std::endl; }private:int m_cMember;
};

上面的代码中:

  • 父类A中,有属性m_aMember,普通函数func1() func2(),有虚函数 v_func1() v_func2()
  • 父类B中,有属性m_bMember,普通函数func1(),虚函数 v_func1() v_func2() v_func4()
  • class C继承了A B,有属性m_cMember,普通函数func1(),虚函数 v_func1() v_func2() v_func3()

在多继承情况下,有多少个基类就有多少个虚函数表指针,前提是基类要有虚函数。
如图,虚函数表指针01指向的虚函数表是以A的虚函数表为基础的,子类C的虚函数vfunc1() vfunc2() 函数指针覆盖了虚函数表01中的虚函数指针01的位置、02位置。

当子类有多出来的虚函数时,会被添加在第一个虚函数表中,所以,子类C的 v_func3 会被添加到以A虚函数表为基础的第一个虚表中。

当有多个虚函数表时,虚函数表的结果是0代表没有下一个虚函数表。" * "号位置在不同操作系统中实现不同,代表有下一个虚函数表

规则:

  1. 子类虚函数会覆盖每一个父类每一个同名虚函数
  2. 父类中没有的虚函数而子类有,填入第一个虚函数表中
  3. 父类中有,而子类中没有,则不覆盖。
int main(int argc, char* argv[])
{class A* a = new class C;a->func1();a->func2();a->v_func1();a->v_func2();class B* b = new class C;b->func1();b->v_func1();b->v_func2();b->v_func4();return 0;
}

运行结果:


文章参考与<零声教育>的C/C++linux服务期高级架构系统教程学习:

C++多态之虚函数表详解及代码示例相关推荐

  1. java集合框架的结构_集合框架(Collections Framework)详解及代码示例

    简介 集合和数组的区别: 数组存储基础数据类型,且每一个数组都只能存储一种数据类型的数据,空间不可变. 集合存储对象,一个集合中可以存储多种类型的对象.空间可变. 严格地说,集合是存储对象的引用,每个 ...

  2. 三维空间刚体运动1:旋转矩阵与变换矩阵(详解加代码示例)

    三维空间刚体运动1:旋转矩阵与变换矩阵(详解加代码示例) 1. 点.向量和坐标系 2.坐标系间的欧式变换 2.1 旋转 2.2 平移 3.齐次坐标和变换矩阵 4. 相似.仿射和射影变换 4.1 相似变 ...

  3. wait 和 waitpid 详解及代码示例

    wait 和 waitpid 详解及代码示例 1. 父子进程处理历史及父进程处理方法 2. wait 2.1 wait 功能 2.2 wait 接口 2.3 wait 原理 2.3.1 wait 源码 ...

  4. Java 泛型(generics)详解及代码示例、Java 类型通配符详解及代码示例

    Java 泛型(generics)详解及代码示例.Java 类型通配符详解及代码示例 - 概念 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制 ...

  5. C++中的多态——理解虚函数表及多态实现原理

    多态及其实现原理 一.多态的概念 概念 构成条件 二.虚函数的重写 重写的定义 重写的特殊情况 override和final关键字 区分重写.重载.重定义 抽象类的概念 三.多态的实现原理 父类对象模 ...

  6. pdo mysql limit_PHP mysql中limit用法详解(代码示例)

    在MySQL中,LIMIT子句与SELECT语句一起使用,以限制结果集中的行数.LIMIT子句接受一个或两个offset和count的参数.这两个参数的值都可以是零或正整数. offset:用于指定要 ...

  7. Pylon SDK的C语言使用流程详解及代码示例

    目录 前言 1.Pylon SDK简介与基本运行流程 2.载入相机 3.流抓取器抓取对象 4.单帧或连续抓图过程 5.收尾工作 5.1卸载流抓取器 5.2卸载相机对象 前言 笔者采用的Pylon版本为 ...

  8. html渐变线条代码,css3线性渐变语法的详解(代码示例)

    本篇文章给大家带来的内容是css3线性渐变语法的详解(代码示例).有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所帮助. 线性渐变的完整语法:.demo { background: line ...

  9. Huffman 编码原理详解(代码示例)

    1.概述 huffman编码是一种可变长编码(  VLC:variable length coding))方式,于1952年由huffman提出.依据字符在需要编码文件中出现的概率提供对字符的唯一编码 ...

最新文章

  1. itemchanged信号找不到_失物 | 求FDU同学帮转帮找蓝牙键盘,坐标东区宿舍19号楼...
  2. scroll-view——小程序横向滚动
  3. C# 11 预览,又增加了实用的语法糖
  4. .NET手撸绘制TypeScript类图——上篇
  5. jzoj5223-B【矩阵乘法】
  6. 使用offsetof对结构体指针偏移操作
  7. LeetCode 1381. 设计一个支持增量操作的栈(deque/数组)
  8. C语言课后习题(10)
  9. 开源串口调试助手java_(串口通信编程) 开源串口调试助手Common (Com Monitor)
  10. ros 发布信息频率_ROS 消息发布器和订阅器Publisher, Subscriber
  11. 通过设置proxyTable实现调用接口跨域
  12. 2018-2019-2 20165313 《网络对抗技术》 Exp6 信息搜集与漏洞扫描
  13. CS 635: Advanced Systems Programming
  14. matlab图像分割(肺实质)
  15. Oracle 锁详解(lock)
  16. C语言常用库函数实现(一)_内存拷贝
  17. LCD1602和12864简单的介绍
  18. 游戏金币单位换算管理类
  19. python名片管理系统_用python实现名片管理系统
  20. inventor铸件图_inventor三维剖视图

热门文章

  1. 中国全干光缆行业市场供需与战略研究报告
  2. 百度网盘cookie_你一定没用过的百度网盘下载神器补充说明
  3. 3-1 郭老师的冰糖葫芦 c++
  4. Too-big precision 7 specified for 'CREATE_TIME'. Maximum is 6.
  5. 【MyCat】mycat简介
  6. java显示1到n的整数值的平方
  7. 牛客杂记——java中左移运算符<<、右移运算符>>和无符号右移运算符>>>的区别
  8. 程序员视频学习网站,推荐这几个网站不错
  9. Vue响应式原理(含详细代码)
  10. Java里什么是POJO