文章目录
1.单继承中的虚函数表整体代码用程序打印虚表如何寻找到虚表地址虚表存在哪里? 2.多继承中的虚函数表整体代码寻找虚表地址注意事项多继承重写后的func1地址为什么不同?ptr1调用函数——一次jmpptr2 调用函数——多次jmp
1.单继承中的虚函数表
整体代码
#include<iostream>using namespace std;class Base{public: virtual void Func1() { cout << "Base::Func1()" << endl; } virtual void Func2() { cout << "Base::Func2()" << endl; } void Func3() { cout << "Base::Func3()" << endl; }private: int _b = 1;};class Derive : public Base{public: virtual void Func1() { cout << "Derive::Func1()" << endl; } virtual void Func4() { cout << "Base::Func4()" << endl; }private: int _d = 2;};typedef void(*VF_PTR)(); // typedef void(*)() VF_PTR;void PrintVFTable(VF_PTR table[])//函数指针数组{ int i = 0; for (i = 0; table[i] != nullptr; i++) { printf("[%d]:%p\n",i, table[i]); VF_PTR f = table[i]; f(); }}int main(){ Base b; Derive d; PrintVFTable((VF_PTR*)*(int*)&b); cout << endl; PrintVFTable((VF_PTR*)*(int*)&d); return 0;}
在子类中实现一个虚函数Func4,但是不构成重写
data:image/s3,"s3://crabby-images/b71f3/b71f3a0af3982b0b372e14fff5aa9806d9157db8" alt=""
data:image/s3,"s3://crabby-images/77083/770837a13749efef581469bbad7380270d04483f" alt=""
Func4函数并没有进入虚表中
data:image/s3,"s3://crabby-images/34d4d/34d4d5a54f9a3220def0a545356299e8baf45a68" alt=""
通过查询内存发现,虚表指针中存在三个地址,而其中两个正好为监视中的两个地址
猜测 0x00c4146a 就是Func4的地址
用程序打印虚表
虚表本质是一个函数指针数组
data:image/s3,"s3://crabby-images/fec1a/fec1ae573c1edb14a6758e0566760c31db1635ca" alt=""
VS中在数组最后放了一个nullptr,这样就可以解决在不同虚表中的个数不同的问题
data:image/s3,"s3://crabby-images/79a3f/79a3f92f7b52d1f5d87534c9d4e7b89ec607cee5" alt=""
typedef一个函数指针 为VF_PTR
正常来说 要写成将VF_PTR放在后面
data:image/s3,"s3://crabby-images/7952e/7952ea59f69ff05a5a390140c98af4150d6f22de" alt=""
但是由于函数指针的特殊性,定义变量都要放在中间
如何寻找到虚表地址
想要虚表的地址,就可以通过找到虚表的指针
而这个虚表指针在对象中,这个指针在对象的前4个(32位)或者8个字节(64位)上面
data:image/s3,"s3://crabby-images/cd390/cd39011583782e2e7b9c19db0ffda73c98e6c07d" alt=""
以32位为例,如何取对象的前4个字节
强制转换为int*
data:image/s3,"s3://crabby-images/ab7b5/ab7b515ce3e96f93b3a861fc04291cac6d08dee4" alt=""
* (int* )&b
首先取到Base* ,将其强制转换为int*,代表前四个字节的地址,再解引用是int类型,把前四个字节取出来
但是由于PrintVFTable函数参数是 函数指针数组
(VF_PTR*) * (int *)&b
如果这个数组是int类型,就需要 一个int * 指针去指向
同理 ,该数组为 VF_PTR类型,需要一个VF_PTR *指针去指向
所以需将 int 类型 再次强制转换为 VF_PTR * ,使其指向这个数组
缺陷
但是这种写法具有一定的局限性,只能在32位跑,因为32位指针大小为4个字节
而64位下就不行了,64位下指针大小为8个字节
运行程序打印虚表,确实了解到多了一个地址
data:image/s3,"s3://crabby-images/19de1/19de1ee82350ac63c6f194043588a5d711e9f564" alt=""
data:image/s3,"s3://crabby-images/6ac53/6ac538aa49954979e4da479701301a7de8406467" alt=""
把虚表的地址拿出来赋给函数指针,用函数指针去调用函数
这里发现 监视中没有出现的地址确实是Func4函数的地址
虚表存在哪里?
data:image/s3,"s3://crabby-images/98f4d/98f4d47cba3bfc00681438130f45aaf17f73ff74" alt=""
由于常量区地址与虚表的地址最为接近,所以说明虚表在常量区/代码段上
2.多继承中的虚函数表
整体代码
class Base1 {public: virtual void func1() { cout << "Base1::func1" << endl; } virtual void func2() { cout << "Base1::func2" << endl; }private: int b1;};class Base2 {public: virtual void func1() { cout << "Base2::func1" << endl; } virtual void func2() { cout << "Base2::func2" << endl; }private: int b2;};class Derive : public Base1, public Base2 {public: virtual void func1() { cout << "Derive::func1" << endl; } virtual void func3() { cout << "Derive::func3" << endl; }private: int d1;};typedef void(*VF_PTR)(); // typedef void(*)() VF_PTR;void PrintVFTable(VF_PTR table[])//函数指针数组{ int i = 0; for (i = 0; table[i] != nullptr; i++) { printf("[%d]:%p->", i, table[i]); VF_PTR f = table[i]; f(); }}int main(){ Derive d; PrintVFTable( (VF_PTR*) *(int*) &d); cout << endl; /*PrintVFTable((VF_PTR*)*(int*)( (char*)&d+ sizeof(Base1) ) );*/ Base2* ptr2 = &d; PrintVFTable((VF_PTR*)*(int*)(ptr2)); return 0;}
寻找虚表地址
data:image/s3,"s3://crabby-images/276ac/276acf65caa8db5ce61aa559458b8c2a9715d388" alt=""
Derive 作为Base1 和Base2的子类,所以Derive内部有两张虚表
data:image/s3,"s3://crabby-images/f13b9/f13b9b451b2e17be317336a261a1a6d091fe6f2e" alt=""
正常来说,Derive内部还存在一个func3函数,这个函数放在哪里了呢?
借助打印虚表来查看,这里的打印虚表依旧可以使用单继承中的那个
data:image/s3,"s3://crabby-images/8675e/8675e62f700fd323bfba7e2ad062b9dc35fefb0c" alt=""
data:image/s3,"s3://crabby-images/cdd50/cdd50b3f7f561474d3679dc62157d806f0faf532" alt=""
base1的虚表指针 正好在对象的前4个字节处,直接可以使用求出虚表指针 去指向base1的虚表
方法1 : base2的虚表指针 需要加上base1的大小
data:image/s3,"s3://crabby-images/5111a/5111a69d6528bcf242b150b437c6b15a2905b443" alt=""
但是这里要注意一个问题,若写成 PrintVFTable((VF_PTR*)(int)( &d+ sizeof(Base1) ) )
写的并不对,d本身是一个Derive类型,&d后变为Derive* 的一个指针,+1 跳转的是Derive类型的字节大小
而该设计想要每次+1跳转1个字节,所以需要强制转换char*
方法2 :切片自动偏移
data:image/s3,"s3://crabby-images/ee51b/ee51ba04435524f32044943f77267040fe6f8d60" alt=""
data:image/s3,"s3://crabby-images/e4219/e4219b169a9094922b8ba941efa44263b12b9073" alt=""
两种方法的结果都是一样的
注意事项
data:image/s3,"s3://crabby-images/11f39/11f39d2160ab9f67aca6b01e8ddfa7ee687d00e3" alt=""
多继承派生类增加的的虚函数在第一个虚表中
多继承重写后的func1地址为什么不同?
data:image/s3,"s3://crabby-images/dba2b/dba2b7a49b05f173eeda9de48d5120e92f309dc0" alt=""
ptr1调用函数——一次jmp
data:image/s3,"s3://crabby-images/b2a3f/b2a3f0e62d056145b73c6bd278cbba2f3f228f2b" alt=""
找到 Base1虚表里的地址 0x00e21230 ,再call这个地址
只需要jmp一次 就可以找到实际真正执行的函数地址
ptr1调用地址属于正常调用
ptr2 调用函数——多次jmp
data:image/s3,"s3://crabby-images/e1174/e1174b105daa4693f8e88f31b0fe3c1f2d852961" alt=""
ptr2调用地址,需要 多次jmp 才能找到真正的函数地址
data:image/s3,"s3://crabby-images/efb9c/efb9c163c8b8cb5448844b8c39fbbe0f73668797" alt=""
ecx存的是this指针
ecx,8 目的是修正this指针的位置
最终Base1和Base2都是执行同一个函数的指令
data:image/s3,"s3://crabby-images/92a40/92a40e56839f802eb80fa019646989e0b2958bcc" alt=""