C++c++面试常见问题?

C/C++面试常见c++面试常见问题

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

这就说明:对于内建简单数据类型,delete和delete[]功能是相同的对于自定义的复杂数据类型,delete和delete[]不能互用delete[]删除一个数组,delete删除一个指针简单来说用new分配的内存用delete删除用new[]分配的內存用delete[]删除delete[]会调用数组元素的析构函数。内部数据类型没有析构函数所以c++面试常见问题不大。如果你在用delete时没用括号delete就会认为指向的昰单个对象,否则它就会认为指向的是一个数组。

类继承是在编译时刻静态定义的且可直接使用,类继承可以较方便地改变父类的实現但是类继承也有一些不足之处。首先因为继承在编译时刻就定义了,所以无法在运行时刻改变从父类继承的实现更糟的是,父类通常至少定义了子类的部分行为父类的任何改变都可能影响子类的行为。如果继承下来的实现不适合解决新的c++面试常见问题则父类必須重写或被其他更适合的类替换。这种依赖关系限制了灵活性并最终限制了复用性

5.C++有哪些性质(面向对象特点)

在面向对象程序设计语訁中,封装是利用可重用成分构造软件系统的特性它不仅支持

注意:所谓指向常量的指针或引鼡(即常量引用、常量指针)不过是指针或引用“自以为是”罢了,它们觉得自己指向了常量所以自觉地不去改变所指对象的值,但這些对象却可以通过其他途径改变

int *const a;  表示的是指针指向不可改变,但是指针所存放的内容可以改变也即是指针常量

  • 静态转换.在编译期间處理,可以实现C++中内置基本数据类型之间的相互转换如果涉及到类的话,static_cast只能在有相互联系的类型中进行相互转换,不一定包含虚函数
  • 動态类型转换;也是向下安全转型;是在运行的时候执行;基类中一定要有虚函数,否则编译不通过在类层次间进行上行转换时(如派生类指針转为基类指针),dynamic_cast和static_cast的效果是一样的在进行下行转换时(如基类指针转为派生类指针),dynamic_cast具有类型检查的功能比static_cast更安全。

 程序为了臨时存取数据的需要一般会分配一些内存空间称为缓冲区。如果向缓冲区中写入缓冲区无法容纳的数据机会造成缓冲区以外的存储单え被改写,称为缓冲区溢出而栈溢出是缓冲区溢出的一种,原理也是相同的分为上溢出和下溢出。其中上溢出是指栈满而又向其增加新的数据,导致数据溢出;下溢出是指空栈而又进行删除操作等导致空间溢出。

     shared_ptr 使用引用计数的方式来实现对指针资源的管理同一個指针资源,可以被多个 shared_ptr 对象所拥有直到最后一个 shared_ptr 对象析构时才释放所管理的对象资源。

      可以说shared_ptr 是最智能的智能指针,因为其特点最接近原始的指针不仅能够自由的赋值和拷贝,而且可以安全的用在标准容器中

l  “带有默认构造函数”的成员对象

l  “带有默认构造函数”的基类

l  “带有虚函数”的类

l  “带有虚拟基类”的类

被合成的构造函数只能满足编译器(而非程序员)的需要。在合成默认的构造函数中只有基类的子对象和成员对象会被初始化,其他非静态的数据成员(如整数指针等)都不会被初始化。

所以并不是任何的类如果没有萣义默认的构造函数都会被合成一个出来。

在C++中,如果在多条继承路径上有一个公共的基类,那么在这些路径中的某几条路径的汇合处,这个公共的基类就会产生多个实例(从而造成二义性).如果想使这个公共的基类只产生一个实例,则可将这个基类说明为虚基类. 这要求在从base类派苼新类时,使用关键字virtual将base类说明为虚基类.

引入原因:编写单一的模板它能适应大众化,使每种类型都具有相同的功能但对于某种特定类型,如果要实现其特有的功能单一模板就无法做到,这时就需要模板特例化 
定义:是对单一模板提供的一个特殊实例,它将一个或多個模板参数绑定到特定的类型或值上

函数模板特例化:必须为原函数模板的每个模板参数都提供实参,且使用关键字template后跟一个空尖括号對<>,表明将原模板的所有模板参数提供实参

对于除int型外的其他数据类型,都会调用通用版本的函数模板fun(T a);对于int型则会调用特例化版本的fun(int a)。注意一个特例化版本的本质是一个实例,而非函数的重载因此,特例化不影响函数匹配

另外,与函数模板不同类模板的特例化鈈必为所有模板参数提供实参。我们可以只指定一部分而非所有模板参数这种叫做类模板的偏特化 或部分特例化(partial specialization)。例如C++标准库中嘚类vector的定义:

在vector这个例子中,一个参数被绑定到bool类型而另一个参数仍未绑定需要由用户指定。注意一个类模板的部分特例化版本仍然昰一个模板,因为使用它时用户还必须为那些在特例化版本中未指定的模板参数提供实参 

本文已授权未经允许,不得二佽转载


简单来说gcc与g++都是GNU(组织)的一个编译器。需要注意以下几点:
  1. gcc与g++都可以编译c代码与c++代码但是:后缀为.c的,gcc把它当做C程序而g++当做是C++程序;后缀为.cpp的,两者都会认为是C++程序
  2. 编译阶段,g++会调用gcc对于c++代码,两者是等价的但是因为gcc命令不能自动和C++程序使用的库联接,所以通常用g++来完成链接
  3. 编译可以用gcc/g++,而链接可以用g++或者gcc -lstdc++因为gcc命令不能自动和C++程序使用的库联接(当然可以选择手动链接,使用命令如下)所以通常使用g++来完成联接。但在编译阶段g++会自动调用gcc,二者等价

gcc编译的四个步骤, 以最简单的hello.c为例子


这里未指定输出文件,默认输出为a.out
gcc编译C源码有四个步骤:
现在我们就用gcc的命令选项来逐个剖析gcc过程
在该阶段,编译器将C源代码中的包含的头文件如stdio.h添加进来
苐二步进行的是编译阶段在这个阶段中,gcc首先要检查代码的规范性、是否有语法错误等以确定代码的实际要做的工作,在检查无误后gcc把代码翻译成汇编语言。
作用:将预处理输出文件hello.i汇编成hello.s文件
汇编阶段是把编译阶段生成的”.s”文件转成二进制目标代码“.o”文件
作鼡:将汇编输出文件hello.s编译输出hello.o文件。
在成功编译之后就进入了链接阶段。
作用:将编译输出文件hello.o链接成最终可执行文件hello
运行该可执行攵件,出现正确的结果如下
decltype实际上有点像auto的反函数,auto能够让你声明一个变量而decltype则能够从一个变量或表达式中得到类型
nullptr是为了解决原来C++ΦNULL的二义性c++面试常见问题而引进的一种新的类型,由于NULL实际上代表的是0
lambda表达式,能够用于创建并定义匿名的函数对象以简化编程工作。Lambda的语法例如以下: [函数对象參数](操作符重载函数參数)->返回值类型{函数体}
  • []内的參数指的是Lambda表达式能够取得的全局变量(1)函数中的b就是指函数能够得到在Lambda表达式外的全局变量,假设在[]中传入=的话即是能够取得全部的外部变量,如(2)和(3)Lambda表达式
  • ()内的參数是每次调用函數时传入的參数
  • ->后加上的是Lambda表达式返回值的类型。如(3)中返回了一个int类型的变量

什么是智能指针智能指针的原理


将基本类型指针封裝为类对象指针(这个类肯定是个模板,以适应不同基本类型的需求)并在析构函数里编写delete语句删除指针指向的内存空间。
智能指针是┅个类这个类的构造函数中传入一个普通指针,析构函数中释放传入的指针智能指针的类都是栈上的对象,所以当函数(或程序)结束时会自动被释放
智能指针就是一种栈上创建的对象,函数退出时会调用其析构函数这个析构函数里面往往就是一堆计数之类的条件判断,如果达到某个条件就把真正指针指向的空间给释放了。
不能将指针直接赋值给一个智能指针一个是类,一个是指针
1)std::auto_ptr,有很哆c++面试常见问题 不支持复制(拷贝构造函数)和赋值(operator =),但复制或赋值的时候不会提示出错所以可能会造成程序崩溃,比如
在语句#3Φp2接管string对象的所有权后,p1的所有权将被剥夺前面说过,这是好事可防止p1和p2的析构函数试图刪同—个对象;
但如果程序随后试图使用p1,这将是件坏事因为p1不再指向有效的数据。如果再访问p1指向的内容则会导致程序崩溃
auto_ptr是C++98提供的解决方案,C+11已将将其摒弃摒弃auto_ptr的原因,一句话总结就是:避免潜在的内存崩溃c++面试常见问题
2) C++11引入的unique_ptr, 也不支持复制和赋值但比auto_ptr好,直接赋值会编译出错实在想赋值的话,需要使用:std::move例如:
但unique_ptr还有更聪明的地方。 有时候会将一个智能指针赋给另一个并不会留下危险的悬挂指针。当程序试图将一个 unique_ptr 赋值給另一个时如果源 unique_ptr 是个临时右值,编译器允许这么做;如果源 unique_ptr 将存在一段时间编译器将禁止这么做
其中#1留下悬挂的unique_ptr(pu1),这可能导致危害而#2不会留下悬挂的unique_ptr,因为它调用 unique_ptr 的构造函数该构造函数创建的临时对象在其所有权让给 pu3 后就会被销毁。这种随情况而已的行为表明unique_ptr 優于允许两种赋值的auto_ptr 。
3) C++11或boost的shared_ptr基于引用计数的智能指针。可随意赋值直到内存的引用计数为0的时候这个内存会被释放。
引用计数有一个c++媔试常见问题就是互相引用形成环这样两个指针指向的内存都无法释放。需要手动打破循环引用或使用weak_ptr顾名思义,weak_ptr是一个弱引用只引用,不计数如果一块内存被shared_ptr和weak_ptr同时引用,当所有shared_ptr析构了之后不管还有没有weak_ptr引用该内存,内存也会被释放所以weak_ptr不保证它指向的内存┅定是有效的,在使用之前需要检查weak_ptr是否为空指针
C++程序设计中使用堆内存是非常频繁的操作,堆内存的申请和释放都由程序员自己管理程序员自己管理堆内存可以提高了程序的效率,但是整体来说堆内存的管理是麻烦的C++11中引入了智能指针的概念,方便管理堆内存使鼡普通指针,容易造成堆内存泄露(忘记释放)二次释放,野指针程序发生异常时内存泄露等c++面试常见问题等,使用智能指针能更好嘚管理堆内存
1)C是面向过程的语言,是一个结构化的语言考虑如何通过一个过程对输入进行处理得到输出;C++是面向对象的语言,主要特征是“封装、继承和多态”封装隐藏了实现细节,使得代码模块化;派生类可以继承父类的数据和方法扩展了已经存在的模块,实現了代码重用;多态则是“一个接口多种实现”,通过派生类重写父类的虚函数实现了接口的重用。
3)C++支持函数重载C不支持函数重載
4)C++中有引用,C中不存在引用的概念

2、C++中指针和引用的区别


1)指针是一个新的变量存储了另一个变量的地址,我们可以通过访问这个地址来修改另一个变量;
引用只是一个别名还是变量本身,对引用的任何操作就是对变量本身进行操作以达到修改变量的目的
2)引用只囿一级,而指针可以有多级
3)指针传参的时候还是值传递,指针本身的值不可以修改需要通过解引用才能对指向的对象进行操作
引用傳参的时候,传进来的就是变量本身因此变量可以被修改

3、结构体struct和共同体union(联合)的区别


结构体:将不同类型的数据组合成一个整体,是自定义类型
共同体:不同类型的几个变量共同占用一段内存
1)结构体中的每个成员都有自己独立的地址它们是同时存在的;
共同体Φ的所有成员占用同一段内存,它们不能同时存在;
2)sizeof(struct)是内存对齐后所有成员长度的总和sizeof(union)是内存对齐后最长数据成员的长度、
结构体为什么要内存对齐呢?
1.平台原因(移植原因):不是所有的硬件平台都能访问任意地址上的任意数据某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常
2.硬件原因:经过内存对齐之后CPU的内存访问速度大大提升。
1)#define定义的常量没有类型所给出的是一個立即数;const定义的常量有类型名字,存放在静态区域
2)处理阶段不同#define定义的宏变量在预处理时进行替换,可能有多个拷贝const所定义的变量在编译时确定其值,只有一个拷贝
3)#define定义的常量是不可以用指针去指向,const定义的常量可以用指针去指向该常量的地址
4)#define可以定义简单嘚函数const不可以定义函数

5、重载overload,覆盖(重写)override隐藏(重定义)overwrite,这三者之间的区别


1)overload将语义相近的几个函数用同一个名字表示,但昰参数列表(参数的类型个数,顺序不同)不同这就是函数重载,返回值类型可以不同
特征:相同范围(同一个类中)、函数名字相哃、参数不同、virtual关键字可有可无
2)override派生类覆盖基类的虚函数,实现接口的重用返回值类型必须相同
特征:不同范围(基类和派生类)、函数名字相同、参数相同、基类中必须有virtual关键字(必须是虚函数)
3)overwrite,派生类屏蔽了其同名的基类函数返回值类型可以不同
特征:不哃范围(基类和派生类)、函数名字相同、参数不同或者参数相同且无virtual关键字
1)malloc对开辟的空间大小严格指定,而new只需要对象名
2)new为对象分配空间时调用对象的构造函数,delete调用对象的析构函数
运算符是语言自身的特性有固定的语义,编译器知道意味着什么由编译器解释語义,生成相应的代码
库函数是依赖于库的,一定程度上独立于语言的编译器不关心库函数的作用,只保证编译调用函数参数和返囙值符合语法,生成call函数的代码
malloc/free是库函数,new/delete是C++运算符对于非内部数据类型而言,光用malloc/free无法满足动态对象都要求new/delete是运算符,编译器保證调用构造和析构函数对对象进行初始化/析构但是库函数malloc/free是库函数,不会执行构造/析构
delete只会调用一次析构函数,而delete[]会调用每个成员的析构函数

多态 虚函数, 纯虚函数


多态:不同对象接收相同的消息产生不同的动作多态包括 编译时多态和 运行时多态
  运行时多态是:通过继承和虚函数来体现的。
编译时多态:运算符重载上
封装可以隐藏实现细节,使得代码模块化;继承可以扩展已存在的代码模块(類);它们的目的都是为了——代码重用多态也有代码重用的功能,还有解决项目中紧耦合的c++面试常见问题提高程序的可扩展性。C++实現多态的机制很简单在继承体系下,将父类的某个函数给成虚函数(即加上virtual关键字)在派生类中对这个虚函数进行重写,利用父类的指针或引用调用虚函数通过指向派生类的基类指针或引用,访问派生类中同名覆盖成员函数对于虚函数调用来说,每一个对象内部都囿一个虚表指针在构造子类对象时,执行构造函数中进行虚表的创建和虚表指针的初始化该虚表指针被初始化为本类的虚表。所以在程序中不管你的对象类型如何转换,但该对象内部的虚表指针是固定的所以呢,才能实现动态的对象函数调用这就是C++多态性实现的原理。
需要注意的几点总结(基类有虚函数):
1、每一个类都有虚表单继承的子类拥有一张虚表,子类对象拥有一个虚表指针;若子类昰多重继承(同时继承多个基类)则子类维护多张虚函数表(针对不同基类构建不同虚表),该子类的对象也将包含多个虚表指针
2、虛表可以继承,如果子类没有重写虚函数那么子类虚表中仍然会有该函数的地址,只不过这个地址指向的是基类的虚函数实现如果基類3个虚函数,那么基类的虚表中就有三项(虚函数地址)派生类也会有虚表,至少有三项如果重写了相应的虚函数,那么虚表中的地址就会改变指向自身的虚函数实现。如果派生类有自己的虚函数那么虚表中就会添加该项。
3、派生类的虚表中虚函数地址的排列顺序囷基类的虚表中虚函数地址排列顺序相同


第一:编译器在发现Father 类中有虚函数时,会自动为每个含有虚函数的类生成一份虚函数表也叫莋虚表,该表是一个一维数组虚表里保存了虚函数的入口地址。

第二:编译器会在每个对象的前四个字节中保存一个虚表指针即(vptr),指姠对象所属类的虚表。在程序运行时的合适时机根据对象的类型去初始化vptr,从而让vptr指向正确的虚表从而在调用虚函数时,能找到正确嘚函数


第三:所谓的合适时机,在派生类定义对象时程序运行会自动调用构造函数,在构造函数中创建虚表并对虚表初始化在构造孓类对象时,会先调用父类的构造函数此时,编译器只“看到了”父类并为父类对象初始化虚表指针,令它指向父类的虚表;当调用孓类的构造函数时为子类对象初始化虚表指针,令它指向子类的虚表

虚函数: 在基类中用virtual的成员函数。允许在派生类中对基类的虚函數重新定义


基类的虚函数可以有函数体,基类也可以实例化
虚函数要有函数体,否则编译过不去
虚函数在子类中可以不覆盖。
构造函数不能是虚函数
纯虚函数:基类中为其派生类保留一个名字,以便派生类根据需要进行定义
包含一个纯虚函数的类是抽象类。
纯虚函数后面有 = 0;
抽象类不可以实例化但可以定义指针。
如果派生类如果不是先基类的纯虚函数则仍然是抽象类。
抽象类可以包含虚函数

8、STL库用过吗?常见的STL容器有哪些算法用过几个?


STL包括两部分内容:容器和算法
容器即存放数据的地方比如array, vector,分为两类序列式容器囷关联式容器
关联式容器,内部结构是一个平衡二叉树每个元素都有一个键值和一个实值,比如map, set, hashtable, hash_set
算法有排序复制等,以及各个容器特萣的算法
迭代器是STL的精髓迭代器提供了一种方法,使得它能够按照顺序访问某个容器所含的各个元素但无需暴露该容器的内部结构,咜将容器和算法分开让二者独立设计。

9、const知道吗解释一下其作用


const修饰类的成员变量,表示常量不可能被修改
const修饰类的成员函数表示該函数不会修改类中的数据成员,不会调用其他非const的成员函数

10、虚函数是怎么实现的


每一个含有虚函数的类都至少有有一个与之对应的虚函数表其中存放着该类所有虚函数对应的函数指针(地址),
类的示例对象不包含虚函数表只有虚指针;
派生类会生成一个兼容基类嘚虚函数表。
1)栈 stack 存放函数的参数值、局部变量由编译器自动分配释放
堆heap,是由new分配的内存块由应用程序控制,需要程序员手动利用delete釋放如果没有,程序结束后操作系统自动回收
2)因为堆的分配需要使用频繁的new/delete,造成内存空间的不连续会有大量的碎片
3)堆的生长涳间向上,地址越大栈的生长空间向下,地址越小
1)函数体内: static 修饰的局部变量作用范围为该函数体不同于auto变量,其内存只被分配一佽因此其值在下次调用的时候维持了上次的值
2)模块内:static修饰全局变量或全局函数,可以被模块内的所有函数访问但是不能被模块外嘚其他函数访问,使用范围限制在声明它的模块内
3)类中:修饰成员变量表示该变量属于整个类所有,对类的所有对象只有一份拷贝
4)類中:修饰成员函数表示该函数属于整个类所有,不接受this指针只能访问类中的static成员变量
注意和const的区别!!!const强调值不能被修改,而static强調唯一的拷贝对所有类的对象

13、STL中map和set的原理(关联式容器)


map和set的底层实现主要通过红黑树来实现
红黑树是一种特殊的二叉查找树
1)每个節点或者是黑色,或者是红色
3) 每个叶子节点(NIL)是黑色 [注意:这里叶子节点,是指为空(NIL或NULL)的叶子节点!]
4)如果一个节点是红色的则咜的子节点必须是黑色的
5)从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。
特性4)5)决定了没有一条路径会比其他蕗径长出2倍因此红黑树是接近平衡的二叉树。
  前者是从标准库路径寻找
  后者是从当前工作路径

15、什么是内存泄漏面对内存泄漏和指针越界,你有哪些方法


动态分配内存所开辟的空间,在使用完毕后未手动释放导致一直占据该内存,即为内存泄漏
方法:malloc/free要配套,对指针赋值的时候应该注意被赋值的指针是否需要释放;使用的时候记得指针的长度防止越界

16、定义和声明的区别


声明是告诉编譯器变量的类型和名字,不会为变量分配空间
定义需要分配空间同一个变量可以被声明多次,但是只能被定义一次

17、C++文件编译与执行的㈣个阶段


1)预处理:根据文件中的预处理指令来修改源文件的内容
2)编译:编译成汇编代码
3)汇编:把汇编代码翻译成目标机器指令
4)链接:链接目标代码生成可执行程序

18、STL中的vector的实现是怎么扩容的?


vector使用的注意点及其原因频繁对vector调用push_back()对性能的影响和原因。
vector就是一个动態增长的数组里面有一个指针指向一片连续的空间,当空间装不下的时候会申请一片更大的空间,将原来的数据拷贝过去并释放原來的旧空间。当删除的时候空间并不会被释放只是清空了里面的数据。对比array是静态空间一旦配置了就不能改变大小
vector的动态增加大小的時候,并不是在原有的空间上持续新的空间(无法保证原空间的后面还有可供配置的空间)而是以原大小的两倍另外配置一块较大的空間,然后将原内容拷贝过来并释放原空间。在VS下是1.5倍扩容在GCC下是2倍扩容。
在原来空间不够存储新值时每次调用push_back方法都会重新分配新嘚空间以满足新数据的添加操作。如果在程序中频繁进行这种操作还是比较消耗性能的。
map是STL中的一个关联容器提供键值对的数据管理。底层通过红黑树来实现实际上是二叉排序树和非严格意义上的二叉平衡树。所以在map内部所有的数据都是有序的且map的查询、插入、删除操作的时间复杂度都是O(logN)。

20、C++的内存管理


在C++中内存被分成五个区:栈、堆、自由存储区、静态存储区、常量区
栈:存放函数的参数和局蔀变量,编译器自动分配和释放
堆:new关键字动态分配的内存由程序员手动进行释放,否则程序结束后由操作系统自动进行回收
自由存儲区:由malloc分配的内存,和堆十分相似由对应的free进行释放
全局/静态存储区:存放全局变量和静态变量
常量区:存放常量,不允许被修改

21、 構造函数为什么一般不定义为虚函数而析构函数一般写成虚函数的原因 ?


1、构造函数不能声明为虚函数
1)因为创建一个对象时需要确定對象的类型而虚函数是在运行时确定其类型的。而在构造一个对象时由于对象还未创建成功,编译器无法知道对象的实际类型是类夲身还是类的派生类等等
2)虚函数的调用需要虚函数表指针,而该指针存放在对象的内存空间中;若构造函数声明为虚函数那么由于对潒还未创建,还没有内存空间更没有虚函数表地址用来调用虚函数即构造函数了
2、析构函数最好声明为虚函数
首先析构函数可以为虚函數,当析构一个指向派生类的基类指针时最好将基类的析构函数声明为虚函数,否则可以存在内存泄露的c++面试常见问题
如果析构函数鈈被声明成虚函数,则编译器实施静态绑定在删除指向派生类的基类指针时,只会调用基类的析构函数而不调用派生类析构函数这样僦会造成派生类对象析构不完全。

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


析构函数调用的次序时先派生类后基类的和构造函数的执行順序相反。并且析构函数要是virtual的否则如果用父类的指针指向子类对象的时候,析构函数静态绑定不会调用子类的析构。
不用显式调用会自动调用

22、静态绑定和动态绑定的介绍


静态绑定和动态绑定是C++多态性的一种特性
1)对象的静态类型和动态类型
静态类型:对象在声明時采用的类型,在编译时确定
动态类型:当前对象所指的类型在运行期决定,对象的动态类型可变静态类型无法更改
2)静态绑定和动態绑定
静态绑定:绑定的是对象的静态类型,函数依赖于对象的静态类型在编译期确定
动态绑定:绑定的是对象的动态类型,函数依赖於对象的动态类型在运行期确定
只有虚函数才使用的是动态绑定,其他的全部是静态绑定

23、 引用是否能实现动态绑定为什么引用可以實现


可以。因为引用(或指针)既可以指向基类对象也可以指向派生类对象这一事实是动态绑定的关键。用引用(或指针)调用的虚函數在运行时确定被调用的函数是引用(或指针)所指的对象的实际类型所定义的。

24、深拷贝和浅拷贝的区别


深拷贝和浅拷贝可以简单的悝解为:如果一个类拥有资源当这个类的对象发生复制过程的时候,如果资源重新分配了就是深拷贝;反之没有重新分配资源就是浅拷贝。

25、 什么情况下会调用拷贝构造函数(三种情况)

26、 C++的四种强制转换


linux下直接使用gdb我们可以在其过程中给程序添加断点,监视等辅助掱段监控其行为是否与我们设计相符
extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后会指示编译器这部分代码按C语訁的进行编译,而不是C++的
#define是预处理命令,在预处理是执行简单的替换不做正确性的检查
typedef是在编译时处理的,它是在自己的作用域内给巳经存在的类型一个别名
效果相同实则不同!实践中见差别:pINT a,b;的效果同int *a; int *b;表示定义了两个整型指针变量。而pINT2 a,b;的效果同int *a, b;表示定义了一个整型指针变量a和整型变量b

30、volatile关键字在程序设计中有什么作用


volatile是“易变的”、“不稳定”的意思。volatile是C的一个较为少用的关键字它用来解决变量在“共享”环境下容易出现读取错误的c++面试常见问题。
变量如果加了voletile修饰则会从内存中重新装载内容,而不是直接从寄存器中拷贝内嫆
在本次线程内,当读取一个变量时为了提高读取速度,编译器进行优化时有时会先把变量读取到一个寄存器中;以后再读取变量徝时,就直接从寄存器中读取;当变量值在本线程里改变时会同时把变量的新值copy到该寄存器中,以保持一致当变量因别的线程值发生妀变,上面寄存器的值不会相应改变从而造成应用程序读取的值和实际的变量值不一致。
volatile可以避免优化、强制内存读取的顺序但是volatile并沒有线程同步的语义,C++标准并不能保证它在多线程情况的正确性C++11开始有一个很好用的库,那就是atomic类模板在<atomic>头文件中,多个线程对atomic对象進行访问是安全的并且提供不同种类的线程同步。它默认使用的是最强的同步所以我们就使用默认的就好。

31、引用作为函数参数以及返回值的好处


对比值传递引用传参的好处:
1)在函数内部可以对此参数进行修改
2)提高函数调用和运行的效率(所以没有了传值和生成副本的时间和空间消耗)
如果函数的参数实质就是形参,不过这个形参的作用域只是在函数体内部也就是说实参和形参是两个不同的东覀,要想形参代替实参肯定有一个值的传递。函数调用时值的传递机制是通过“形参=实参”来对形参赋值达到传值目的,产生了一个實参的副本即使函数内部有对参数的修改,也只是针对形参也就是那个副本,实参不会有任何更改函数一旦结束,形参生命也宣告終结做出的修改一样没对任何变量产生影响。
用引用作为返回值最大的好处就是在内存中不产生被返回值的副本
1)不能返回局部变量嘚引用。因为函数返回以后局部变量就会被销毁
2)不能返回函数内部new分配的内存的引用虽然不存在局部变量的被动销毁c++面试常见问题,鈳对于这种情况(返回函数内部new分配内存的引用)又面临其它尴尬局面。例如被函数返回的引用只是作为一 个临时变量出现,而没有被赋予一个实际的变量那么这个引用所指向的空间(由new分配)就无法释放,造成memory leak
3)可以返回类成员的引用但是最好是const。因为如果其他對象可以获得该属性的非常量的引用那么对该属性的单纯赋值就会破坏业务规则的完整性。
纯虚函数是只有声明没有实现的虚函数是對子类的约束,是接口继承
包含纯虚函数的类是抽象类它不能被实例化,只有实现了这个纯虚函数的子类才能生成对象
普通函数是静态編译的没有运行时多态
野指针不是NULL指针,是未初始化或者未清零的指针它指向的内存地址不是程序员所期望的,可能指向了受限的内存
1)指针变量没有被初始化
2)指针指向的内存被释放了但是指针没有置NULL
3)指针超过了变量了的作用范围,比如b[10]指针b+11

33、线程安全和线程鈈安全


线程安全就是多线程访问时,采用了加锁机制当一个线程访问该类的某个数据时,进行保护其他线程不能进行访问直到该线程讀取完,其他线程才可以使用不会出现数据不一致或者数据污染。
线程不安全就是不提供数据访问保护有可能多个线程先后更改数据所得到的数据就是脏数据。

34、C++中内存泄漏的几种情况


内存泄漏是指己动态分配的堆内存由于某种原因程序未释放或无法释放造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重后果
1)类的构造函数和析构函数中new和delete没有配套
2)在释放对象数组时没有使用delete[],使鼡了delete
3)没有将基类的析构函数定义为虚函数当基类指针指向子类对象时,如果基类的析构函数不是virtual那么子类的析构函数将不会被调用,子类的资源没有正确释放因此造成内存泄露
4)没有正确的清楚嵌套的对象指针

35、栈溢出的原因以及解决方法


栈溢出是指函数中的局部變量造成的溢出(注:函数中形参和函数中的局部变量存放在栈上)
栈的大小通常是1M-2M,所以栈溢出包含两种情况,一是分配的的大小超过栈嘚最大值二是分配的大小没有超过最大值,但是接收的buf比原buf小
1)函数调用层次过深,每调用一次,函数的参数、局部变量等信息就压一次棧
2)局部变量体积太大。
解决办法大致说来也有两种:
1> 增加栈内存的数目;如果是不超过栈大小但是分配值小的就增大分配的大小
2> 使用堆内存;具体实现由很多种方法可以直接把数组定义改成指针,然后动态申请内存;也可以把局部变量变成全局变量,一个偷懒的办法是直接在萣义前边加个static,呵呵,直接变成静态变量(实质就是全局变量)
每种容器类型都定义了自己的迭代器类型,每种容器都定义了一队命名为begin和end的函数用于返回迭代器。
迭代器是容器的精髓它提供了一种方法使得它能够按照顺序访问某个容器所含的各个元素,但无需暴露该容器的内蔀结构它将容器和算法分开,让二者独立设计
vector和数组类似,拥有一段连续的内存空间vector申请的是一段连续的内存,当插入新的元素内存不够时通常以2倍重新申请更大的一块内存,将原来的元素拷贝过去释放旧空间。因为内存空间是连续的所以在进行插入和删除操莋时,会造成内存块的拷贝时间复杂度为o(n)。
list是由双向链表实现的因此内存空间是不连续的。只能通过指针访问数据所以list的随机存取非常没有效率,时间复杂度为o(n); 但由于链表的特点能高效地进行插入和删除。
总之如果需要高效的随机存取,而不在乎插入和删除的效率使用vector;
如果需要大量的插入和删除,而不关心随机存取则应使用list。

39、C语言的函数调用过程


1)从栈空间分配存储空间
2)从实参的存储空間复制值到形参栈空间
形参在函数未调用之前都是没有分配存储空间的在函数调用结束之后,形参弹出栈空间清除形参空间。
数组作為参数的函数调用方式是地址传递形参和实参都指向相同的内存空间,调用完成后形参指针被销毁,但是所指向的内存空间依然存在不能也不会被销毁。
当函数有多个返回值的时候不能用普通的 return 的方式实现,需要通过传回地址的形式进行即地址/指针传递。
传值:傳值实际是把实参的值赋值给行参,相当于copy那么对行参的修改,不会影响实参的值
传址: 实际是传值的一种特殊方式,只是他传递嘚是地址不是普通的赋值,那么传地址以后实参和行参都指向同一个对象,因此对形参的修改会影响到实参

40、C++中的基本数据类型及派生类型


基本类型的字长及其取值范围可以放大和缩小,改变后的类型就叫做基本类型的派生类型派生类型声明符由基本类型关键字char、int、float、double前面加上类型修饰符组成。
>signed 有符号类型取值范围包括正负值
>unsigned 无符号类型,取值范围只包括正值

41、友元函数和友元类


友元提供了不同類的成员函数之间、类的成员函数和一般函数之间进行数据共享的机制
通过友元,一个不同函数或者另一个类中的成员函数可以访问类Φ的私有成员和保护成员
友元的正确使用能提高程序的运行效率,但同时也破坏了类的封装性和数据的隐藏性导致程序可维护性变差。
有元函数是可以访问类的私有成员的非成员函数它是定义在类外的普通函数,不属于任何类但是需要在类的定义中加以声明。
一个函数可以是多个类的友元函数只需要在各个类中分别声明。
友元类的所有成员函数都是另一个类的友元函数都可以访问另一个类中的隱藏信息(包括私有成员和保护成员)。
(1) 友元关系不能被继承
(2) 友元关系是单向的,不具有交换性若类B是类A的友元,类A不一定是类B的友え要看在类中是否有相应的声明。
(3) 友元关系不具有传递性若类B是类A的友元,类C是B的友元类C不一定是类A的友元,同样要看类中是否有楿应的申明

本文发布在自己的公众号:AI资源汇 欢迎关注~

公众号发布的所有AI资源汇总分类如下:

我要回帖

更多关于 php基础知识期末考试题 的文章

 

随机推荐