虚函数的一个小小的测试~

[入库:2005年8月19日] [更新:2007年3月25日]

本文简介:选择自 oury 的 blog

虚函数的一个小小的测试~
#include <iostream>

using namespace std;

 

class a

{

public:

   virtual void fun(){cout<<1<<endl;}

   virtual void fun2(){cout<<2<<endl;}

};

 

class b:public a

{

public:

   int xx;

   void fun(){cout<<3<<endl;}

   void fun2(){cout<<4<<endl;}

};

 

int main()

{

1.  void (*fun)(a*);

2.  a *p=new b;

3.  long lvptraddr;

4.  memcpy(&lvptraddr,p,4);

5.  memcpy(&fun,reinterpret_cast<long*>(lvptraddr),4);

6.  fun(p);

7.  delete p;

  system("pause");

}

 

 解释:
1. 声明一个函数指针,用于保存vtable中的函数地址.
2. 声明一个基类的指针,指向b.
3. 定义一个long 以保存地址.
4. 进行内存的拷贝,把p指向的vptr的地址拷贝到lvptraddr里,做为一个long 值,即:取得vptr的地址.
5. 因为vptr指向vtable,所以reinterpret_cast<long*>(lvptraddr)转换成指针,取得指针所指的vtable的地址.拷贝给fun.
6.fun(p)调用具体的virtual函数,由于this指针的存在,所以需要加上p.做为参数.

我想肯定有人会这么想,为什么需要类型的转换?
   原因也很简单,如果直接*p,可以得到vptr的地址么?不会,只会得到属于p指向的一个对象.只有把它转换一下,才能取得真正的指针值.
(我说的不是很清楚(才疏学浅),希望看到的网友能帮我说说清楚,谢谢.)

当然3、4、5可以合并成一句:memcpy(&fun,reinterpret_cast<long*>(*reinterpret_cast<long*>(p)))
它的意思: 把p转换成long*,然后再取它的值;其实转换后的值也是一个指针(vptr),vptr的值当然就是vtable的地址值,把vtable的地址赋值fun(它就直接指向了vtable中的第一个虚函数).最终得到vtable的地址。

说明:
   这个程序只能测试vtable中函数的另类调用方法,并不能说明虚函数的工作原理,因为这里有一句“a *p=new b;”实际在我们想象中,它已经可以试图调用虚函数,但我们并没有直接用p->fun()或p->fun2();而是用另类的方法调用。

但不知道如果这样行不行:

 

int main()

{

1.  void (*fun)();

2.  a *p=new b;

3.  long lvptraddr;

4.  memcpy(&lvptraddr,p,4);

5.  memcpy(&fun,reinterpret_cast<long*>(lvptraddr),4);

6.  fun();

7.  delete p;

  system("pause");

}

请说出为什么?谢谢。

本文关键:虚函数的一个小小的测试~
 

本站最佳浏览方式为 分辨率 1024x768 IE 6.0(或更高版本的 IE浏览器)

go top