虚函数的一个小小的测试~
来源:互联网 发布:淘宝客拍a发b2017 编辑:程序博客网 时间:2024/05/18 03:36
虚函数的一个小小的测试~
#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");
}
请说出为什么?谢谢。
- 虚函数的一个小小的测试~
- dom4j 的小小测试
- 小小的测试题
- 测试的小小收获
- 一个小小的烦恼
- 一个小小的错误
- 一个小小的问题
- 一个小小的问题
- 一个小小的Slider~
- 一个小小的计算器
- 一个小小的dp
- 一个小小的进步~
- 一个小小的makefile
- 一个小小的案例
- 一个小小的项目
- 关于创建主键和索引的关系一个小小测试
- 一个小小的测试题,求解(欢迎解答)
- 测试开发之路--一个小小工程师的回首
- 如何让你的SQL运行得更快
- 浅谈C++函数的参数
- 关于指针和内存的几个问题
- 指针的引用与正常指针 释放内存的另一例比较
- 多维静态、动态数组存贮与访问的讨论
- 虚函数的一个小小的测试~
- 关于memcpy的用法
- 宽字符输出
- 什么时候应该使用 ==?什么时候应该使用 Equals?
- 我眼中的指针--学习指针不可少的好文章
- 指针函数和函数指针有什么区别
- JPEG 简易文档 V2.14
- 关于C++模板和重载的小问题
- C++异常处理