C++的虚函数的作用就是为了实现多态的机制,利用内存的指针偏移来实现将基类型的指针指向的内存空间用子类对象来初始化。这样经过内部虚表的运作,实现可以通过基类指针来调用子类所定义的方法。

这种技术,其实就是一种泛型技术,试图让不变的代码来实现可变的算法。比如:模板、RTTI、虚函数。实现在编译时决定方法,或者就是在运行时决定方法

虚函数表:

简称vtbl,这个表中保存了一个类的虚函数地址,这张表解决了继承、覆盖的问题。

在有虚函数的类对象中这个表被分配在对象的内存空间中。创建虚表,赋值地址的工作,其实都是由编译器来完成,编译器通过自动添加代码到构造函数中来实现。

C++的编译器应该保证虚函数表的指针存在于对象实例内存中最前面的位置。既然是首地址,那么当我们获取一个对象的首地址时,其实也就是得到了这个对象所使用的虚表指针的地址,于是也就可以顺利的得到虚表的地址。

 class Base
{
public:
virtual void f() {cout << "Base::f()" << endl; };
virtual void g() {cout << "Base::g()" << endl; };
virtual void h() {cout << "Base::h()" << endl; };
};
// 按照上面的类声明,我们来创建一个对象
// 然后得到这个对象所使用的虚表
Base b;
typedef void (*Fun)(void);
Fun pFun = NULL; cout << "虚函数表地址" << (int*)(&b) << endl;
// 其实就是虚指针vbtr的值,因为vbtr的地址和对象的地址相同 cout << "虚函数表: 第一个函数地址" << (int*)*(int*)(&b) << endl; pFun = (Fun)*((int*)*(int*)(&b));
pFun();

运行结果:

然后通过: pFun = (Fun)* ((int*)*(int*)(&b) + 1);取得第二个函数, pFun = (Fun)* ((int*)*(int*)(&b) + 2); 取得第三个函数;

以此类推....

一般继承(无虚函数覆盖):

通过一个子类Derive继承基类,但不做虚函数覆盖,于是子类的虚表如下所示:

 class Derive : public Base
{
public:
virtual void f() {cout << "Derive::f()" << endl; };
virtual void g() {cout << "Derive::g()" << endl; };
virtual void h() {cout << "Derive::h()" << endl; };
virtual void f1() {cout << "Derive::f1()" << endl; };
virtual void g1() {cout << "Derive::g1()" << endl; };
virtual void h1() {cout << "Derive::h1()" << endl; };
protected:
private:
};

子类虚函数表:***Base::f()地址***Base::g()地址***Base::h()地址***Derive::f1()地址***Derive::g1()地址***Derive::h1()地址***

作为子类对象,它的虚表和Base类的虚表不是同一个表,子类的虚函数如果不实现,则以此排列。

一般继承(有虚函数覆盖):

 class Derive : public Base
{
public:
void f() {cout << "Derive::f()" << endl; };
protected:
private:
};

此时Derive的虚表,第1个函数已经指向了子类的实现。

多重继承(无虚函数覆盖):

每个父类都有自己的虚表,但子类成员函数被放在了第一个声明基类的虚表中。

多重继承(有虚函数覆盖):

三个基类表中的f()位置都被替换成了子类的函数指针

C++虚函数表(vtbl)的更多相关文章

  1. C++虚函数与虚函数表

    多态性可分为两类:静态多态和动态多态.函数重载和运算符重载实现的多态属于静态多态,动态多态性是通过虚函数实现的. 每个含有虚函数的类有一张虚函数表(vtbl),表中每一项是一个虚函数的地址, 也就是说 ...

  2. [转] C++虚函数与虚函数表

    http://www.cnblogs.com/Ripper-Y/archive/2012/05/15/2501930.html http://blog.csdn.net/haoel/article/d ...

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

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

  4. C++迟后联编和虚函数表

    先看一个题目: class Base { public: virtual void Show(int x) { cout << "In Base class, int x = & ...

  5. C++对象内存模型2 (虚函数,虚指针,虚函数表)

    从例子入手,考察如下带有虚函数的类的对象内存模型: class A { public: virtual void vfunc1(); virtual void vfunc2(); void func1 ...

  6. C++对象的内存分布和虚函数表

    c++中一个类中无非有四种成员:静态数据成员和非静态数据成员,静态函数和非静态函数. 1.非静态数据成员被放在每一个对象体内作为对象专有的数据成员.    2.静态数据成员被提取出来放在程序的静态数据 ...

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

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

  8. C++虚函数表与虚析构函数

    1.静态联编和动态联编联编:将源代码中的函数调用解释为要执行函数代码. 静态联编:编译时能确定唯一函数.在C中,每个函数名都能确定唯一的函数代码.在C++中,因为有函数重载,编译器须根据函数名,参数才 ...

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

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

随机推荐

  1. js中的变量提升(hoisting)

    来看如下代码: function HelloJS(){ var array = [1,2,3,4,5]; for(var i in array){ } alert(i); } HelloJS(); a ...

  2. linux 配置Tomcat开机启动

    一台安装有tomcat的linux服务器 方法/步骤   1 请自行下载安装配置tomcat的服务器环境 本经验仅仅介绍如何配置tomcat的开机自动启动 2 切换到tomcat/bin目录下 用vi ...

  3. Java GC的原理

    Java GC(garbage collec,垃圾收集,回收) GC是对JVM中的内存进行标记和回收,Sun公司的JDK用的虚拟机都是HotSpot 对象化的实例是放在heap堆内存中的,这里讲的分代 ...

  4. thymeleaf从session中获取数据

    <input th:value="${session.value1}" />

  5. HTTP Error 500.0 - Internal Server Error错误代码0x80070002

    案例研究:AspNetInitClrHostFailureModule中的“HTTP错误500.0 - 内部服务器错误” 症状 当用户访问在Windows Server 2008 R2计算机上运行的A ...

  6. JPA报错, java.lang.NullPointerException

    java.lang.NullPointerException 我觉得这应该是一个很常见的错误, 数据库没取到数据嘛, 很正常, JPA没取到数据就是会抛出空指针异常, 但是就是这么简单的一个错误也让我 ...

  7. #define宏常量和const常量的区别

    C++ 语言可以用const 来定义常量,也可以用#define 来定义常量.但是前者比后者有更多的优点:(1) const 常量有数据类型,而宏常量没有数据类型.编译器可以对前者进行类型安全检查.而 ...

  8. CSS文字大小单位px、em、pt详解

    这里引用的是Jorux的“95%的中国网站需要重写CSS”的文章,题目有点吓人,但是确实是现在国内网页制作方面的一些缺陷.我一直也搞不清楚px与em之间的关系和特点,看过以后确实收获很大.平时都是用p ...

  9. 归并排序/合并排序c++实现

    #include <iostream>#include<string.h> using namespace std;class merges{public:void merge ...

  10. WCF生成客户端代理对象的两种方法的解释

    最近在封装WCF,有一些很好的实践就记录下来,大家可以放心使用,所有代码都已经调试过.如果有高手可以大家探讨一下. 在WCF中有两种不同的方法可以用于创建客户端服务对象,他们分别为: 1. 代理构造法 ...