VPTR与构造函数和继承

原创 2007年09月11日 20:03:00
    C++中类的成员函数默认情况下是non-virtual,即被调用时为静态绑定。
   
   
至少包含一个virtual成员函数的类,都有一个VTABLE——虚函数映射表,表中的每项对应类中一个virtual成员函数的函数体地址。相应的该类的每个对象在为其分配存储空间时,编译器会额外的为每个对象附加一个指针VPTR,该指针指向该对象所属类的VTABLE。

    一定要明确概念,VTABLE是在类这个层次上的概念,而VPTR则是在对象这个层次上的概念。

    将VPTR正确设置、指向合适的VTABLE,这是由谁负责完成的?类的构造函数。编译器会自动的在构造函数中插入设置VPTR的代码。

    常见的实现中,编译器会将VPTR放在对象所占空间的头部。

    对于单重继承关系中的子类对象,其VPTR的设置则经历如下两个阶段

    A).首先,在基类构造函数中,VPTR被设置为指向基类的VTABLE
    B).之后,在子类构造函数中,VPTR被设置为指向子类的VTABLE

    注意,这里只存在一个VPTR,在子对象的构建过程中被重写。

    而对于多重继承,有类似的过程,不过子类对象中存在不止一个VPTR。

下面是一个简单的验证代码,可以观察VPTR是如何被重写,以及指向何处

#include <iostream>

using namespace std;

class Base
{
public:
    
int x;
    Base():x(
0)
    
{
    
void * pv=this;
    
int *  pi=static_cast<int *>(pv);
    printf(
"vptr in base ctor point to : %x ",*pi);

    }

    
virtual ~Base(){}
}
;

class Derived:public Base
{
public:
    Derived()
    
{

    
void * pv=this;
    
int *  pi=static_cast<int *>(pv);
    printf(
"vptr in derived  point to : %x ",*pi);
    }


}
;


int main(int argc, char* argv[])
{

    Base Ba;
    Derived Da;

    
void * pv=&Ba;
    
int * pi=static_cast<int *>(pv);
    printf(
"address of Base's vTable  : %x ",(*pi));


    pv
=&Da;
    pi
=static_cast<int *>(pv);
    printf(
"address of Derived's vTable  : %x ",(*pi));

    
return 1;
}


运行结果

vptr in base ctor point to :    8048af0

vptr 
in base ctor point to :    8048af0     //initialized by base ctor
vptr in derived  point to :     8048b10     //overwritten by derived ctor 

address of Base
's vTable  :     8048af0
address of Derived's vTable  :  8048b10

相关文章推荐

C/C++—— 在构造函数中调用虚函数能实现多态吗(Vptr指针初始化的过程分析)

问题引入:比如:如果我们想在父类的构造函数中调用虚函数,当定义子类对象的时候,父类的构造函数中的虚函数执行的是子类中的函数。...

Java继承时构造函数的调用

  • 2013年08月06日 15:30
  • 22KB
  • 下载

继承类的构造函数

  • 2014年10月24日 17:22
  • 48KB
  • 下载

C#构造函数的声明、重载与继承调用

今天小智终于开始写C#的博客了,c语言属于面向过程的编程语言,而c#则属于面向对象的编程语言,相信看此片博客的同伴们应该已经了解面向对象以及他的三大特性:多态,封装,继承了吧,今天小智对此也不做太多介...

类的继承和构造函数

  • 2014年04月03日 15:56
  • 869B
  • 下载

构造函数继承.关于java

  • 2011年03月30日 21:35
  • 4KB
  • 下载

C#静态构造函数,在继承中调用情况

public class A{ public static string strText; public string Text; static A()...

Java继承时构造函数的调用

  • 2007年09月25日 02:49
  • 4KB
  • 下载

构造函数运算符重载

  • 2016年03月04日 14:59
  • 31KB
  • 下载

C++中继承中遇到的构造函数问题

今天在开发时遇到了一个之前一直以为理所当然的构造函数问题。 先给总结: 子类在构造时,如果没有显式调用父类的构造函数,会先调用父类的默认构造函数(无参数的) 下面给出不同情况的例子 例...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:VPTR与构造函数和继承
举报原因:
原因补充:

(最多只允许输入30个字)