OS:Windows 7

关键字:VS2015,C++,V-Table,虚表,虚函数。

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

单一继承情况:

#include "stdafx.h"
#include <string>
#include <iostream> using namespace std; class Base
{
public:
virtual std::string Name();
}; std::string Base::Name()
{
return "Base";
} class A : public Base
{
public:
std::string Name() override;
virtual std::string AName();
}; std::string A::Name()
{
return AName();
} std::string A::AName()
{
return "A";
} int main()
{
cout << "Size of Base: " << sizeof(Base) << endl;
cout << "Size of A: " << sizeof(A) << endl; Base* pBase = new Base();
cout << pBase->Name() << endl;
A* pA = new A();
cout << pA->Name() << endl;
Base* pBaseA = pA;
cout << pBaseA->Name() << endl; return ;
}

控制台输出:

x86:

Size of Base: 4
Size of A: 4
Base
A
A

x64:

Size of Base:
Size of A:
Base
A
A

虚表结构:

总结:

  • Base和A对象的大小都是一个虚表指针的大小。一个指针大小在x86 (32bit)程序里是4个字节,在x64(64bit)程序里面是8个字节。
  • Watch窗口没有显示虚表里面的AName,在Watch立面添加“(*((Base*)pA)).__vfptr[1]”可以看到。

多重继承情况:

#include "stdafx.h"
#include <string>
#include <iostream> using namespace std; class A
{
public:
virtual string AName() { return "A"; }
}; class B
{
public:
virtual string BName() { return "B"; }
}; class C : public A, public B
{
public:
virtual string BName() { return "C"; }
}; int main(int argc, _TCHAR* argv[])
{
cout << "Size of A: " << sizeof(A) << endl;
cout << "Size of B: " << sizeof(B) << endl;
cout << "Size of C: " << sizeof(C) << endl;
C* pC = new C();
C* pC1 = new C();
cout << pC->BName() << endl;
B* pB = static_cast<B*>(pC);
cout << pB->BName() << endl;
//A* pA = static_cast<A*>(pB);
A* pA = dynamic_cast<A*>(pB);
cout << pA->AName() << endl;
A* pA1 = reinterpret_cast<A*>(pB);
cout << pA1->AName() << endl;
return ;
}

控制台输出:

x86:

Size of A:
Size of B:
Size of C:
C
C
A
C

x64:

Size of A:
Size of B:
Size of C:
C
C
A
C

看到这个结果有没有让你奇怪的地方?为什么“pA1->AName()”输出是C?

虚表结构:

总结:

  • pC和pC1是类C的两个实例指针,这两个实例的虚表是一样的,也就是说虚表是属于类的,一个类有一个虚表。
  • 因为类C重写了类B的BName函数,所以C的虚表里存的是C::BName
  • “B* pB = static_cast<B*>(pC);”,父类的指针来操作一个子类,虚函数实现了多态。
  • “A* pA = static_cast<A*>(pB);”是编译不通过的,因为A和B是不相关的两个类,也就是没有继承关系。
  • “A* pA = dynamic_cast<A*>(pB);”是可以的,因为dynamic_cast会在运行时进行类型检查,dynamic_cast是最安全的,但是效率最低。
  • “A* pA1 = reinterpret_cast<A*>(pB);”是可以编译通过了,但是运行时会有意想不到的结果。reinterpret_cast进行了强制类型转换,但是不会纠正虚表。
  • 派生类C有两个虚表指针分别指向两个虚表。

虚拟继承情况:

#include "stdafx.h"
#include <string>
#include <iostream> using namespace std; class Base
{
public:
virtual string Name();
}; string Base::Name()
{
return "Base";
} class A : virtual public Base
{
public:
string Name() override;
virtual string AName();
}; string A::Name()
{
return AName();
return "A";
} string A::AName()
{
return "A";
} int main()
{
cout << "Size of Base: " << sizeof(Base) << endl;
cout << "Size of A: " << sizeof(A) << endl; Base* pBase = new Base();
cout << pBase->Name() << endl;
A* pA = new A();
cout << pA->Name() << endl;
Base* pBaseA = pA;
cout << pBaseA->Name() << endl; return ;
}

控制台输出:

x86:

Size of Base:
Size of A:
Base
A
A

x64:

Size of Base:
Size of A:
Base
A
A

虚表结构:

总结:

  • 以x64为例,相比非虚单继承,A的大小打了16个字节,是因为A多了一个虚表指针指向用于存放A的虚函数AName的虚表,另外虚继承本身占了8个字节。

实例解析C++虚表的更多相关文章

  1. exec函数族实例解析

    exec函数族实例解析 fork()函数通过系统调用创建一个与原来进程(父进程)几乎完全相同的进程(子进程是父进程的副本,它将获得父进程数据空间.堆.栈等资源的副本.注意,子进程持有的是上述存储空间的 ...

  2. [Reprint] C++函数模板与类模板实例解析

    这篇文章主要介绍了C++函数模板与类模板,需要的朋友可以参考下   本文针对C++函数模板与类模板进行了较为详尽的实例解析,有助于帮助读者加深对C++函数模板与类模板的理解.具体内容如下: 泛型编程( ...

  3. [Reprint]C++普通函数指针与成员函数指针实例解析

    这篇文章主要介绍了C++普通函数指针与成员函数指针,很重要的知识点,需要的朋友可以参考下   C++的函数指针(function pointer)是通过指向函数的指针间接调用函数.相信很多人对指向一般 ...

  4. JavaWeb实现文件上传下载功能实例解析

    转:http://www.cnblogs.com/xdp-gacl/p/4200090.html JavaWeb实现文件上传下载功能实例解析 在Web应用系统开发中,文件上传和下载功能是非常常用的功能 ...

  5. Android实例-Delphi开发蓝牙官方实例解析(XE10+小米2+小米5)

    相关资料:1.http://blog.csdn.net/laorenshen/article/details/411498032.http://www.cnblogs.com/findumars/p/ ...

  6. Android开发之IPC进程间通信-AIDL介绍及实例解析

    一.IPC进程间通信 IPC是进程间通信方法的统称,Linux IPC包括以下方法,Android的进程间通信主要采用是哪些方法呢? 1. 管道(Pipe)及有名管道(named pipe):管道可用 ...

  7. easyUI:ComboTree and comselector使用实例解析

    ComboTree 使用场景:故名思意,ComboTree是combox和Tree的结合体,在需要通过选择得到某一个node值的时候触发. 栗子: 定义: 使用标签创建树形下拉框. Comselect ...

  8. Maven--多模块依赖实例解析(五)

    <Maven--搭建开发环境(一)> <Maven--构建企业级仓库(二)> <Maven—几个需要补充的问题(三)> <Maven—生命周期和插件(四)&g ...

  9. SoapUI简介和入门实例解析

    SoapUI简介 SoapUI是一个开源测试工具,通过soap/http来检查.调用.实现Web Service的功能/负载/符合性测试.该工具既可作为一个单独的测试软件使用,也可利用插件集成到Ecl ...

随机推荐

  1. 360开源的类Redis存储系统:Pika

    Pika 是 360 DBA 和基础架构组联合开发的类 Redis 存储系统,完全支持 Redis 协议,用户不需要修改任何代码,就可以将服务迁移至 Pika.有维护 Redis 经验的 DBA 维护 ...

  2. [Effective C++ --019]设计class犹如设计type

    前言 我们在编写程序的时候,无论哪一种语言,总是会告诉你这种语言应该有的类型.我们将它们直接拿来使用,可是从来没有考虑过这些类到底是怎么设计出来的! 所幸的是:在OOP中,每当我们定义一个新的clas ...

  3. combo参数配置_手册

    combotree : 设置为多选框: $('#menu-combotree').combotree({multiple:true}).combotree('loadData', menuListJs ...

  4. JFinal极速开发实战-业务功能开发-通用表单验证器

    提交表单数据时,需要经过前端的验证才能提交到后台,而后台的验证器再做一道数据的校验,成功之后才能进入action进行业务数据的处理. 在表单数据的验证中,数据类型的验证还是比较固定的.首先是对录入数据 ...

  5. git 撤销修改以及删除文件

    撤销修改 1.如果当你修改了代码,然后又发现修改错误以后,想撤销前面的操作的时候该怎么办呢? 既然错误发现得很及时,就可以很容易地纠正它.你可以删掉最后一行,手动把文件恢复到上一个版本的状态.如果用 ...

  6. 【转载】架构师需要了解的Paxos原理、历程及实战

    原文链接,请参见:http://weibo.com/ttarticle/p/show?id=2309403952892003376258 数据库高可用性难题 数据库的数据一致和持续可用对电子商务和互联 ...

  7. dede常用命令

    获取日期:全局:{dede:field.pubdate function="MyDate('Y-m-d H:i',@me)"/}   局部:[field:pubdate funct ...

  8. JAXB - XML Schema Types, Defining Subtypes

    Although object orientation isn't a key feature of XML or the XML Schema language, it's still possib ...

  9. length prototype 函数function的属性,以及构造函数

    前言:学到一些JavaScript高级的知识,在这里记下,方便以后的查找 1.length代表函数定义的形参的个数,挺简单的     例如:function Pen(price,cname) {  . ...

  10. SQL Server 2012 中 Update FROM子句

    首先说明一下需求以及环境 创建Table1以及Table2两张表,并插入一下数据 USE AdventureWorks2012; GO IF OBJECT_ID ('dbo.Table1', 'U') ...