LeetCode C++基础面试题汇总附答案(二)

目录

前言

面试题及答案

1. 在什么情况下,析构函数需要是虚函数?

2. 基类的析构函数不是虚函数,会带来什么问题?

3. 内联函数、构造函数、静态成员函数可以是虚函数吗?

4. 构造函数中可以调用虚函数吗?

5. 简述 C++ 中虚继承的作用及底层实现原理?

6. new、delete、malloc、free 之间的关系?

7. delete 和delete [] 的区别

8. 子类析构时要调用父类的析构函数吗?

9. 什么是“引用”?申明和使用“引用”要注意哪些问题?

10. 结构与联合有何区别?

11. 重载(overload)和重写(override,有的书也叫做“覆盖”)的区别?


前言

        这里整理了leetCode C++的基础面试第二部分,希望对大家有所帮助 

面试题及答案

1. 在什么情况下,析构函数需要是虚函数?


若存在类继承关系并且析构函数中需要析构某些资源时,析构函数需要是虚函数,否则当使用父类指针指向子类对象,在delete时只会调用父类的析构函数,而不能调用子类的析构函数,造成内存泄露等问题


2. 基类的析构函数不是虚函数,会带来什么问题?


派生类的析构函数用不上,会造成资源的泄漏。


3. 内联函数、构造函数、静态成员函数可以是虚函数吗?


都不可以。内联函数需要在编译阶段展开,而虚函数是运行时动态绑定的,编译时无法展开; 构造函数在进行调用时还不存在父类和子类的概念,父类只会调用父类的构造函数,子类调用子类 的,因此不存在动态绑定的概念;静态成员函数是以类为单位的函数,与具体对象无关,虚函数是 与对象动态绑定的,因此是两个不冲突的概念;


4. 构造函数中可以调用虚函数吗?


可以,但是没有动态绑定的效果,父类构造函数中调用的仍然是父类版本的函数,子类中调用的仍然是子类版本的函数


5. 简述 C++ 中虚继承的作用及底层实现原理?


虚继承用于解决多继承条件下的菱形继承问题,底层实现原理与编译器相关,一般通过虚基类 指针实现,即各对象中只保存一份父类的对象,多继承时通过虚基类指针引用该公共对象,从而避 免菱形继承中的二义性问题。


6. new、delete、malloc、free 之间的关系?


malloc 和 free 都是 C/C++ 语言的标准库函数,new/delete 是 C++ 的运算符。
new 调用构造函数,delete 会调用对象的析构函数,而 free 只会释放内存。
它们都可用于申请动态内存和释放内存。但对于非内部数据类型的对象而言,光用 malloc/free 无法满足动态对象的要求。对象在创建的同时要自动执行构造函数,对象在消亡之前要自动执行析构函数。由于 malloc/free 是库函数而不是运算符,不在编译器控制权限之内,不能够把执行构造函数和析构函数的任务强加给 malloc/free。因此C++语言需要一个能完成动态内存分配和初始化工作的运算符new,以及一个能完成清理与释放内存工作的运算符 delete。注意:new/delete 不是库函数。


7. delete 和delete [] 的区别


delete 只会调用一次析构函数,而 delete[] 会调用每一个成员函数的析构函数。


8. 子类析构时要调用父类的析构函数吗?


析构函数调用的次序是先派生类的析构后基类的析构,也就是说在基类的的析构调用的时候,派生类的信息已经全部销毁了。定义一个对象时先调用基类的构造函数、然后调用派生类的构造函数;析构的时候恰好相反:先调用派生类的析构函数、然后调用基类的析构函数。


9. 什么是“引用”?申明和使用“引用”要注意哪些问题?


引用就是某个目标变量的“别名”,对应用的操作与变量直接操作效果完全相同。声明一个引用的时候,切记要对其进行初始化。引用声明完毕后,相当于目标变量名有两个名称,即该目标原名称和引用名,不能再把该引用名作为其他变量名的别名。声明一个引用,不是新定义了一个变量,它只表示该引用名是目标变量名的一个别名,它本身不是一种数据类型,因为该引用本身不占存储单元,系统也不给引用分配存储单元。不能建立数组的引用。


10. 结构与联合有何区别?


(1). 结构和联合都是由多个不同的数据类型成员组成, 但在任何同一时刻, 联合中只存放了一个被选中的成员(所有成员共用一块地址空间), 而结构的所有成员都存在(不同成员的存放地址不同)。
(2). 对于联合的不同成员赋值, 将会对其它成员重写, 原来成员的值就不存在了, 而对于结构的不同成员赋值是互不影响的。


11. 重载(overload)和重写(override,有的书也叫做“覆盖”)的区别?


从定义上来说:
重载:是指允许存在多个同名函数,而这些函数的参数表不同(或许参数个数不同,或许参数类型不同,或许两者都不同)。
重写:是指子类重新定义父类虚函数的方法。
从实现原理上来说:
重载:编译器根据函数不同的参数表,对同名函数的名称做修饰,然后这些同名函数就成了不同的函数(至少对于编译器来说是这样的)。如,有两个同名函数:function func(p:integer):integer;和function func(p:string):integer;。那么编译器做过修饰后的函数名称可能是这样的:int_func、str_func。对于这两个函数的调用,在编译器间就已经确定了,是静态的。也就是说,它们的地址在编译期就绑定了(早绑定)。
重写:当子类重新定义了父类的虚函数后,父类指针根据赋给它的不同的子类指针,动态的调用属于子类的该函数,这样的函数调用在编译期间是无法确定的(调用的子类的虚函数的地址无法给出)。因此,这样的函数地址是在运行期绑定的(晚绑定)。

上一篇:LeetCode C++基础面试题汇总附答案(一)

下一篇:LeetCode C++基础面试题汇总附答案(三)
参考:

链接:https://leetcode.cn/leetbook/read/da-han-hou-duan-gang-ti-mu-he-ji-shang/nihxf6/
 

猜你喜欢

转载自blog.csdn.net/lejian/article/details/129160723