多态:

C++的多态是通过一张虚函数表(Virtual Table)来实现的,简称为 V-Table。在这个
表中,主要是一个类的虚函数的地址表,这张表解决了继承、覆写的问题,保证其真实
反应实际的函数。这样,在有虚函数的类的实例中这个表被分配在了这个实例的内存中,
所以,当我们用父类的指针来操作一个子类的时候,这张虚函数表就显得由为重要了,
它就像一个地图一样,指明了实际所应该调用的函数

这里我们着重看一下这张虚函数表。C++的编译器应该是保证虚函数表的指针存在
于对象实例中最前面的位置(这是为了保证取到虚函数表的有最高的性能——如果有
多层继承或是多重继承的情况下)。 这意味着我们通过对象实例的地址得到这张虚函
数表,然后就可以遍历其中函数指针,并调用相应的函数

以下代码运行结果,基于环境 X86_64 64位编译器。

虚函数表与与表指针

#include <iostream>

using namespace std;

class Base {
public:
virtual void f() { cout << "Base::f" << endl; }
virtual void g() { cout << "Base::g" << endl; }
virtual void h() { cout << "Base::h" << endl; }
int ba;
}; typedef void(*FUNC)();
int main()
{
Base b;
cout<<sizeof(b)<<endl;
cout<<"Base_Addr:"<<(int*)(&b)<<endl;
cout<<"VTalbe_Addr:"<<(int**)(int*)(&b)<<endl;
FUNC pf = NULL;
pf = (FUNC)*((int**)*(int*)(&b)+0);
pf();
pf = (FUNC)*((int**)*(int*)(&b)+1);
pf();
pf = (FUNC)*((int**)*(int*)(&b)+2);
pf();
}

运行结果:

虚函数表排在一个类的最前面,虚函数在虚函数表中按声明顺序排列。

派生类部分实现覆写:

class Base {
public:
  virtual void f() { cout << "Base::f" << endl; }
  virtual void g() { cout << "Base::g" << endl; }
  virtual void h() { cout << "Base::h" << endl; }
}; class Derived:public Base
{
public:
  virtual void f() { cout<<"Derived::f"<<endl; }
}; typedef void(*FUNC)();
int main()
{
Derived b;
cout<<sizeof(b)<<endl;
cout<<"Base_Addr:"<<(int*)(&b)<<endl;
cout<<"VTalbe_Addr:"<<(int**)(int*)(&b)<<endl;
FUNC pf = NULL;
pf = (FUNC)*((int**)*(int*)(&b)+0);
pf();
pf = (FUNC)*((int**)*(int*)(&b)+1);
pf();
pf = (FUNC)*((int**)*(int*)(&b)+2);
pf();
}

运行结果:

单继承:

父子中兼有虚函数

class Base {
public:
  virtual void f() { cout << "Base::f" << endl; }
  virtual void g() { cout << "Base::g" << endl; }
  virtual void h() { cout << "Base::h" << endl; }
protected:
//int b;
}; class Derived:public Base
{
public:
virtual void d() { cout << "Derived::d" << endl; }

};
typedef void(*FUNC)();
int main()
{
Derived d;
cout<<sizeof(d)<<endl;
cout<<"Base_Addr:"<<(int*)(&d)<<endl;
cout<<"VTalbe_Addr:"<<(int**)(int*)(&b)<<endl;
FUNC pf = NULL;
pf = (FUNC)*((int**)*(int*)(&d)+0);
pf();
pf = (FUNC)*((int**)*(int*)(&d)+1);
pf();
pf = (FUNC)*((int**)*(int*)(&d)+2);
pf(); pf = (FUNC)*((int**)*(int*)(&d)+3);
pf(); Base *pb = & d;
pf = (FUNC)*((int**)*(int*)(pb)+3);
pf();
}

运行结果:

虚函数表形式

 总结:单继承体系中,派生类继承基类,并且有自己独立的虚函数,基类与派生类的虚函数表顺序存放在一张表中,派生类的虚函数表在基类后面,类中的成员变量不影响虚函数表的排布。

多继承:

基类中未完全覆写,派生类中又有虚函数

#include <iostream>

using namespace std;

class Base1 {
public:
virtual void f() { cout << "Base::f" << endl; }
virtual void g() { cout << "Base::g" << endl; }
virtual void h() { cout << "Base::h" << endl; }
}; class Base2 {
public:
virtual void f2() { cout << "Base::f2" << endl; }
virtual void g2() { cout << "Base::g2" << endl; }
virtual void h2() { cout << "Base::h2" << endl; }
}; class Derived:public Base1,public Base2
{
public:
void f() { cout << "Derived::f" << endl; } //override
virtual void i() { cout << "Derived::i" << endl; }
}; typedef void(*FUNC)();
int main()
{
Derived d;
   cout<<sizeof(d)<<endl;
Base1 * p1d = &d;
Base2 * p2d = &d; cout<<"p1d:"<<p1d<<endl;
cout<<"p2d:"<<p2d<<endl; FUNC pf = NULL;
pf = (FUNC)*((int**)*(int*)(p1d)+0);
pf();
pf = (FUNC)*((int**)*(int*)(p1d)+1);
pf();
pf = (FUNC)*((int**)*(int*)(p1d)+2);
pf();
pf = (FUNC)*((int**)*(int*)(p1d)+3);
pf();
   cout<<"-------------------------"<<endl; pf = (FUNC)*((int**)*(int*)(p2d)+0);
pf();
pf = (FUNC)*((int**)*(int*)(p2d)+1);
pf();
pf = (FUNC)*((int**)*(int*)(p2d)+2);
pf();
// pf = (FUNC)*((int**)*(int*)(p2d)+3); //crash// pf();
}

运行结果

继承关系:

虚函数表:

 结论:多继承中,每个基类都有自己的虚函数表,他们是独立存放的,如果派生类中有虚函数,那么虚函数表将与第一个继承的基类虚函数表顺序存放(合并)。

C++多态中虚函数表合并与继承问题的更多相关文章

  1. 关于C++中虚函数表存放位置的思考

    其实这是我前一段时间思考过的一个问题,是在看<深入探索C++对象模型>这本书的时候我产生的一个疑问,最近在网上又看到类似的帖子,贴出来看看: 我看到了很多有意思的答案,都回答的比较好,下面 ...

  2. 【C++ Primer | 15】C++虚函数表剖析①

    概述 为了实现C++的多态,C++使用了一种动态绑定的技术.这个技术的核心是虚函数表(下文简称虚表).本文介绍虚函数表是如何实现动态绑定的. C++多态实现的原理: •  当类中声明虚函数时,编译器会 ...

  3. C++进阶之虚函数表

    C++通过继承(inheritance)和虚函数(virtual function)来实现多态性.所谓多态,简单地说就是,将基类的指针或引用绑定到子类的实例,然后通过基类的指针或引用调用实际子类的成员 ...

  4. C++ 由虚基类 虚继承 虚函数 到 虚函数表

    //虚基类:一个类可以在一个类族中既被用作虚基类,也被用作非虚基类. class Base1{ public: Base1(){cout<<"Construct Base1!&q ...

  5. C++之虚函数表

    本文引自:http://songlee24.github.io/blog/2014/09/02/c-plus-plus-jin-jie-zhi-xu-han-shu-biao/ C++通过继承(inh ...

  6. C++虚函数表和对象存储

    C++虚函数表和对象存储 C++中的虚函数实现了多态的机制,也就是用父类型指针指向其子类的实例,然后通过父类的指针调用实际子类的成员函数,这种技术可以让父类的指针有"多种形态",这 ...

  7. c++基础之虚函数表指针和虚函数表创建时机

    虚函数表指针 虚函数表指针随对象走,它发生在对象运行期,当对象创建的时候,虚函数表表指针位于该对象所在内存的最前面. 使用虚函数时,虚函数表指针指向虚函数表中的函数地址即可实现多态. 虚函数表 虚函数 ...

  8. C++虚函数和虚函数表

    前导 在上面的博文中描述了基类中存在虚函数时,基类和派生类中虚函数表的结构. 在派生类也定义了虚函数时,函数表又是怎样的结构呢? 先看下面的示例代码: #include <iostream> ...

  9. C++ 类的存储方式以及虚函数表

    一.C++成员函数在内存中的存储方式 用类去定义对象时,系统会为每一个对象分配存储空间.如果一个类包括了数据和函数,要分别为数据和函数的代码分配存储空间.按理说,如果用同一个类定义了10个对象,那么就 ...

随机推荐

  1. vue-qiankun公司微前端项稳定目落地后的总结(附github仓库demo,将会持续更新)

    ️本文为博客园社区首发文章,未获授权禁止转载 大家好,我是aehyok,一个住在深圳城市的佛系码农‍♀️,如果你喜欢我的文章,可以通过点赞帮我聚集灵力️. 个人github仓库地址: https:gi ...

  2. Qt5MV自定义模型与实例浅析

    1. Model/View结构 这种结构,其实就是将界面组件与所编辑的数据分离开来,又通过数据源的方式连接起来,相当于解耦,视图层只关心显示和与用户交互,而数据层负责与实际的数据进行通信,并为视图组件 ...

  3. ES6 let const关键字

    在es6中,引入了let和const关键字: 1.letES6 新增了let命令,用来声明变量.它的用法类似于var,但是所声明的变量,只在let命令所在的代码块内有效. (1)在块级作用域里有效(比 ...

  4. Maven作用及应用

    1.简介 Maven是一个项目管理的Java 工具,在JavaEE中,我们可以使用Maven方便地管理团队合作的项目,现在我们在学习JavaEE框架,使用Maven可以管理类库,有效方便地供团队中的其 ...

  5. 将base64Url对应图片保存到本地

    上图中的内容就是base64编码之后对应的Url  图中base64,之前的都是用于声明该图片的格式以及它的编码格式  base64,之后的就是该图片对应的数据了 我们只需要把数据转换为字节保存下来即 ...

  6. python框架之Flask

    介绍:Flask是一个使用 Python 编写的轻量级 Web 应用框架.其 WSGI 工具箱采用 Werkzeug ,模板引擎则使用 Jinja2 . WSGl:Web Server Gateway ...

  7. 第八篇--编写Windows服务

    编写service服务参考网址:https://blog.csdn.net/nodeathphoenix/article/details/24181509 vc获得显示器状态(捕获息屏.亮屏网址):h ...

  8. 开源基于docker的任务调度器pipeline,比`quartzs` 更强大的分布式任务调度器

    pipeline 分布式任务调度器 目标: 基于docker的布式任务调度器, 比quartzs,xxl-job 更强大的分布式任务调度器. 可以将要执行的任务打包为docker镜像,或者选择已有镜像 ...

  9. ifix中嵌入3d模型初探(一)

    在ifix项目中插入3d模型,是当前工控上位机的一个发展趋势,故而我也来尝尝鲜.利用现有条件,初步打算完成一个工厂俯视3d全景. 基本思路:利用webbrowser+3dmax+three.js来嵌入 ...

  10. DC-6靶机

    仅供个人娱乐 靶机信息 下载地址:https://download.vulnhub.com/dc/DC-6.zip 一.主机发现 nmap -sn 192.168.216.0/24 二.端口扫描 nm ...