通过声明对象Base base这种方式去调用virtual函数和调用普通函数没有任何的区别。
#include<stdio.h>
class Base
{
public:
void Function_1()
{
printf("Function_1...\n");
}
virtual void Function_2()
{
printf("Function_2...\n");
}
};
int main(int argc,char * argv[])
{
Base base;
base.Function_1();
base.Function_2();
return 0;
}
通过声明指针Base base这种方式去调用virtual函数和调用普通函数的反汇编有很大区别,不是直接CALL(FF),而是间接CALL(E8)。
int main(int argc,char * argv[])
{
Base base;
Base* pb = &base;
pb->Function_1();
pb->Function_2();
return 0;
}
当前类中没有虚函数时,当前类的大小是类中的成员数据的大小(类中的函数并不占用空间),但一旦加上virtual虚函数后,类的大小会增加4个字节,而且无论加多少虚函数,都是4个字节,实际上就是多出一个数组,地址也是数组的首地址,而数组里面存的是所有虚函数的首地址。(也称为虚函数表)而且这四个字节的地址是类的首地址,This指针的地址,而虚表的地址是this指针的前4个字节。
#include<stdio.h>
class Base
{
public:
virtual void Function_1()
{
printf("Function_1...\n");
}
virtual void Function_2()
{
printf("Function_2...\n");
}
};
int main(int argc,char * argv[])
{
Base base;
printf("base的This只针对的地址为%x\n",&base);
//而虚表是this指针的前4个字节。
printf("base的this前四个字节为%x\n",(int *)&base);
printf("base的虚函数表的地址为%x\n",*(int *)&base);
typedef void(*pFunction)(void);
pFunction pfn;
pfn = (pFunction) *((int *)(*(int *)&base)+0);
pfn();
pfn = (pFunction) *((int *)(*(int *)&base)+1);
pfn();
return 0;
}