现在的位置: 首页 > 综合 > 正文

晒晒C++:虚函数的真相(VC编译器如何实现“virtual ”规则)

2013年10月03日 ⁄ 综合 ⁄ 共 2059字 ⁄ 字号 评论关闭

本文摘自:http://blog.csdn.net/Dreamcode/article/details/7061747#comments

可到 CSDN 下载中心下载全文
http://download.csdn.net/detail/Dreamcode/201005 

(1)virtual 虚函数

先看一段简单代码:

Code Segment:

Line01:  #include<stdio.h>

Line02:

Line03:  class Base {

Line04:  public:

Line05:      virtual void __stdcall Output() {

Line06:         printf("Class Base\n");

Line07:     }

Line08:  };

Line09: 

Line10:  class Derive :public Base {

Line11:  public:

Line12:     void __stdcall Output() {

Line13:         printf("Class Derive\n");

Line14:     }

Line15:  };

Line16: 

Line17:  void Test(Base *p) {

Line18:     p->Output();

Line19:  }

Line20: 

Line21:  int __cdecl main(intargc, char* argv[]) {

Line22:     Derive obj;

Line23:     Test(&obj);

Line24:     return 0;

Line25:  }   

 

基类的“Output”函数是个虚函数。那么,很明显地,程序的运行结果将是:

 

( 2)  virtual function table  虚函数表

先来分析我们的main函数中的Derive类的对象obj,看看它的内存布局,由于没有数据成员,它的大小为4个字节,只有一个vfptr,所以obj的地址也就是vfptr的地址了。

对一个C++类,如果它要呈现多态(一般的编译器会将这个类以及它的基类中是否存在virtual关键字作为这个类是否要多态),那么类会有一个virtual function table,而每一个实例(对象)都会有一个virtual function pointer(以下简称vfptr)指向该类的virtual function table的起始地址,而virtual function table表格地址所对应的内存单元的内容就是虚函数地址(其实并不是真正的函数地址,而是跳转到函数的jmp指令的地址)。

( 2 )  实现 virtual 功能

   

(1)virtual 虚函数

先看一段简单代码:

Code Segment:

Line01:  #include<stdio.h>

Line02:

Line03:  class Base {

Line04:  public:

Line05:      virtual void __stdcall Output() {

Line06:         printf("Class Base\n");

Line07:     }

Line08:  };

Line09: 

Line10:  class Derive :public Base {

Line11:  public:

Line12:     void __stdcall Output() {

Line13:         printf("Class Derive\n");

Line14:     }

Line15:  };

Line16: 

Line17:  void Test(Base *p) {

Line18:     p->Output();

Line19:  }

Line20: 

Line21:  int __cdecl main(intargc, char* argv[]) {

Line22:     Derive obj;

Line23:     Test(&obj);

Line24:     return 0;

Line25:  }   

 

基类的“Output”函数是个虚函数。那么,很明显地,程序的运行结果将是:

 

( 2)  virtual function table  虚函数表

先来分析我们的main函数中的Derive类的对象obj,看看它的内存布局,由于没有数据成员,它的大小为4个字节,只有一个vfptr,所以obj的地址也就是vfptr的地址了。

对一个C++类,如果它要呈现多态(一般的编译器会将这个类以及它的基类中是否存在virtual关键字作为这个类是否要多态),那么类会有一个virtual function table,而每一个实例(对象)都会有一个virtual function pointer(以下简称vfptr)指向该类的virtual function table的起始地址,而virtual function table表格地址所对应的内存单元的内容就是虚函数地址(其实并不是真正的函数地址,而是跳转到函数的jmp指令的地址)。

( 2 )  实现 virtual 功能

   

抱歉!评论已关闭.