1. 什么是虚函数、纯虚函数?
答:
虚函数:类中声明函数时,前面加了virtual关键字的函数。
纯虚函数:末尾加了“=0”的虚函数。
虚函数的作用是为了在使用时后期绑定实现多态。
2. 虚函数是怎样实现的?
答:
简单地说,虚函数是通过虚函数表(vbtl)来实现的,如果一个类中含有虚函数时,那么系统就自动会为该类分配一个指针成员指向一张虚函数表。在这张表中,主要是一个类的虚函数的地址表,这张表解决了继承、覆盖的问题,其内容真实反映实际的函数,在有虚函数的类的实例中,这张表被分配在这个实例的内存中,所以当用父类的指针来操作一个子类时,这个表就指明了实际所调用的函数。
在类对象的内存布局中,首先是类的vbtl指针,然后才是对象数据,所以在通过对象指针调用一个虚函数时,编译器生成的代码将先获取对象类的vbtl指针,然后调用vbtl指针对象的项。对于通过对象指针调用虚函数的情况,在编译期间无法确定指针指向的是基类对象还是哪个派生类对象,但在运行期间执行到该语句时这一点是可以确定的。
3. 一个类实例化后的多个对象中虚函数表是共用的吗?
答:
不同的类具有不同的虚函数表,但如果一个类实例化多个对象,那么这些对象的虚函数表是共用的吗?首先如果我们去设计内存占用时,从节省内存的角度去考虑的话,这些实例化后的类应该是共用一张虚函数表的,因为每个内中的虚函数都是一样的。
测试:
class classA {
virtual void function() {
}
};
int main()
{
classA *a = new classA();
printf("%x\n", *(int*)(void*)a);
for (int i = 0; i < 10000;i++) {
classA *b = new classA();
if (*(int*)(void*)a == *(int*)(void*)b) {
printf("SAME\n");
}
else {
printf("DIFFERENT\n");
break;
}
delete b;
}
return 0;
}
测试结果:满屏的SAME,也就是说:同属一个类的不同的实例化对象是共用一张虚函数表的。

被折叠的 条评论
为什么被折叠?



