C++面试题(三)

1.虚函数表存放的内容
答:虚函数表实际是一个数组,数组里存放的是虚函数的地址指针,指向虚函数。
虚函数表的由来:如果一个类中有虚函数,则每一个该类的对象都会被编译器添加一个成员:虚函数表指针,这个虚函数表指针指向虚函数表,虚函数表本质是一个数组,数组中存放类中虚函数地址指针,指向虚函数。

2.虚函数表指针与虚函数表的存放位置★
答:(当前的答案不一定正确仅供参考)
①虚函数表是class specific的,也就是针对一个类来说的,这里有点像一个类里面的staic成员变量,即它是属于一个类所有对象的,不是属于某一个对象特有的,是一个类所有对象共有的。
②虚函数表是编译器来选择实现的,编译器的种类不同,可能实现方式不一样,就像前面我们说的vptr在一个对象的最前面,但是也有其他实现方式,不过目前gcc 和微软的编译器都是将vptr放在对象内存布局的最前面。
③实际上虚函数指针是在构造函数执行时初始化的,而虚函数表是存放在可执行文件中的。在gcc编译器的实现中虚函数表vtable存放在可执行文件的只读数据段.rodata中

3.构造函数可以是虚函数吗,为什么?
答:不可以,可以从两方面解释:第一,虚函数的作用就是在于信息不全的时候可以重载函数得到对应的调用。构造函数本身就是要初始化实例,那使用虚函数也就没有实际意义。所以构造函数没有必要是虚函数;第二,每一个虚函数对象对应一个虚函数指针,指向该类的一个虚函数表,虚函数指针内存被分配在虚函数对象存储空间的头部,而一个类中虚函数表的创建是在构造函数中被创建的,如果构造函数本身就是虚函数,那在执行构造函数的时候就需要找对应的虚函数指针和虚函数表,而此时虚表指针和虚表还没有被创建,则无法完成查找虚表而报错。
用一个不太恰当的比喻:例如一个显赫的家族(相当于一个类),里面可能有很多人享受着祖上的财富而吃喝玩乐,虚度光阴(虚成员函数),但是在一个家族中,归根到底总要有一个刻苦努力的人来打下基业(创建虚函数表),这个能干有才的人就是构造函数。如果假设这个打基业的人也是一个无所事事的人,那岂不是打了个寂寞,后面的假设也就都不成立了。

4.析构函数可以是虚函数吗,为什么?
答:析构函数可以是虚函数,且一般基类的析构函数均设置为虚函数,如果基类析构函数不为虚函数,那么当new-delete指向子类对象的基类指针时,执行的操作如下:
基类构造函数-子类构造函数-基类的析构函数;
从而导致子类没有执行子类的析构函数而导致内存泄露。
当基类的析构函数被设置为虚函数时,当new-delete指向子类对象的基类指针时,执行的操作如下:
基类构造函数-子类构造函数-子类析构函数-基类析构函数;
这样就保证内存空间不会产生泄露。

你可能感兴趣的:(C++面试题(三))