【C++面试】关于虚函数的常见问题

1.虚函数的代价ios

1)带有虚函数的每一个类会产生一个虚函数表,用来存储虚成员函数的指针c++

2)带有虚函数的每一个类都会有一个指向虚函数表的指针ide

3)再也不是内敛函数,由于内敛函数能够在编译阶段进行替代,而虚函数表示等待,在运行阶段才能肯定到达采用哪一种函数,因此虚函数不是内敛函数函数

 

2.那些函数不能是虚函数?this

1)构造函数:对象的虚函数表指针须要经过构造函数初始化spa

2)内联函数:内联函数能够在编译阶段进行函数体的替换,而虚函数须要在运行期间进行肯定指针

3)静态函数:静态函数不属于对象而属于类,由于静态成员函数没有this指针,因此没法访问对象的虚表指针,也就code

没法访问类的虚函数表,将静态函数设置成虚函数也就没有任何意义,因此c++语法不支持将静态函数设置成虚函数对象

4)友元函数:友元函数不属于类,也不能被继承,没有继承特性的函数没有虚函数的说法blog

5)类外的普通函数:类外普通函数不是类的成员函数,一样不具有继承特性,也就没有虚函数的说法

 

3.虚函数和纯虚函数的区别?

1)纯虚函数只有定义,没有实现,虚函数既有定义,又有实现

2)含有纯虚函数的类不能定义对象,含有虚函数的类能够定义对象

 

4.菱形继承的内存结构?如何解决菱形继承存在的问题?

#include <iostream>
using namespace std; class A { public: int x; virtual int getx() { return x; } }; class B:public A{}; class C:public A{}; class D:public B,public C{}; int main() { D d; //如下代码不会被任何编译器经过,由于存在菱形继承问题
    cout<<d.getx()<<endl; }
View Code

 

1)菱形继承的内存结构:如今有A,B,C,D四个类,B,C分别继承A类,D经过多重继承继承了BC两个类,如今D类中有两个getx(),D类不知道调用哪个getx()

2)菱形继承的解决办法:虚继承

BC类都用Virtual标注,保证只有一个getx()被建立

#include <iostream>
using namespace std; class A { public: int x=5; virtual int getx() { return x; } }; class B:virtual public A{}; class C:virtual public A{}; class D:public B,public C{}; int main() { D d; cout<<d.getx()<<endl; }
View Code

 

5.虚析构函数的做用?父类的析构函数为何必定要设置成虚函数?

父类虚析构函数就是为了避免内存泄漏,防止子类内存得不到释放形成内存泄漏

1.当父类的析构函数不声明成虚析构函数时,当子类继承父类,父类指针指向子类对象,delete掉父类指针,只会调动父类的析构函数,而不会调用子类的析构函数,从而形成子类对象内存泄漏

2.当父类的析构函数声明成虚析构函数时,当子类继承父类,父类指针指向子类对象,delete掉父类指针,先调动父类的析构函数,而后调用子类的析构函数,不存在子类对象内存泄漏的问题

只要存在继承关系,则父类的虚函数必须定义成虚函数!

 

6.构造函数和析构函数中为何不能够调用虚函数?

背景知识:

1.构造子类对象时,首先调用父类构造函数初始化对象的父类部分,在执行父类的构造函数时,对象的子类部分都是未初始化的,实际上此时对象还不是一个子类对象

2.析构子类对象时,先析构子类部分,而后按照构造顺序逆序析构父类部分

因此在运行子类的构造和析构函数时,对象都是不完整的,为了适应这种不完整,编译器视对象类型为当前构造或析构函数所在类的类型,由此形成的结构就是:在父类的构造或析构函数中,会将子类对象看成父类对象看待

在这样的背景下

若是咱们在父类的构造或析构函数中调用虚函数,调用的每每是当前类的虚函数,达不到多态的效果,跟普通函数调用没有区别

 

7.构造函数为何不能为虚函数?什么状况下析构函数必须为虚函数?

1)由于虚函数表指针必须在构造函数中初始化,因此构造函数不能为虚函数!

2)当存在继承关系时,父类的析构函数必须为虚函数,这样在父类指针指向子类对象,delete父类指针时,子类对象才不会内存泄漏

相关文章
相关标签/搜索