1、指针的优点和缺点
优点:灵活高效
(1)提高程序的编译效率和执行速度(数组下标往下移时,需要使用乘法和加法,而指针直接使用++即可)
(2)通过指针可使用主调函数和被调函数之间共享变量或数据结构,便于实现双向数据通讯。
(3)可以实现动态的存储分配。
(4)便于表示各种数据结构,如结构体,编写高质量的程序。
缺点:容易出错
(1)可能变成野指针,导致程序崩溃
(2)内存泄露
(3)可读性差
2、指针和引用的定义和区别
(1)指针和引用的定义
1)指针:指针是一个变量,存储一个地址,指向内存的一个存储单元;
2)引用跟原来的变量实质上是同一个东西,只不过是原变量的一个别名而已。
(2)指针和引用的区别
<1> 从内存分配上来说:
1)指针是一个实体,而引用仅是个别名,即为指针分配内存,而不为引用分配内存空间;
<2> 从指向的内容来说:
2)引用只能在定义时被初始化一次,之后不可变;指针可变;
3)引用不能为空,指针可以为空;
4)const与指针搭配可以表示指针指向和指针指向内容是否可变。const与引用搭配只有一种,即来修饰其内容的可读性。由于引用从一而终,不用修饰其指向。
5)指针可以有多级,但是引用只能是一级(int **p;合法,而int &&a是不合法的)
<3> 其他方面
6)"sizeof引用"得到的是所指向的变量(对象)的大小,而"sizeof指针"得到的是指针本身的大小;
7)指针和引用的自增(++)运算意义不一样;
指针和引用在符号表中的形式:程序在编译时分别将指针和引用添加到符号表上。在符号表上记录的是变量名及变量所对应地址。在符号表上,指针变量对应的地址值为指针变量的地址值,而引用对应的地址值是引用对象的地址值。符号表生成后就不会再改,因此指针可以改变指向的对象(指针变量中的值可以改),而引用对象不能改。
3、malloc/free 和 new/delete相关的面试题
1):C++有了malloc/free 为什么还要new/delete?
(1)malloc/free只在申请空间时,它们只需要申请空间,无法对空间进行操作。
(2)而在创建C++的对象时,不仅仅是需要申请空间,还需要自动调用构造函数,以及在对象消亡之前要自动执行析构函数。
因此 C++语言需要一个能完成动态内存分配和初始化工作的运算符new,以及一个能完成清理与释放内存工作的运算符delete。
根据上面两点,我们可以知道malloc/free 是不能满足C++的需要的,因此需要new/delete。
即在创建对象时,能分配内存空间且初始化内存 + 销毁对象时,能回收空间且对内存进行清理的new/delete。
2)为什么malloc/free在申请空间时,它们只能申请空间,无法对空间进行操作?
因为:malloc/free 是库函数,使用它需要头文件,在一定程度上是独立于语言的。编译器在处理库函数时,编译器不需要知道它是做啥的,而仅仅需要对该函数进行编译,并且保证调用函数时的参数和返回值是合法的,并生成相应 call 函数的代码就ok了。编译器不会控制库函数做一些操作,比如调用构造函数和析构函数。因此malloc/free无法满足动态生成对象的要求。
3)为什么new/delete 在申请空间时,它们不仅能申请空间,还能调用构造函数或析构函数对对空间进行操作?
因为:new/delete是运算符,它与+-*/的地位 是一样的。编译器看到new/delete时,就知道只要它要做啥操作,并生成对应的代码。
4):malloc/free 和 new/delete 的相同点和不同点
相同点:它们都可以申请和释放空间。
不同点:
一、new/delete 在申请空间的时候能对空间进行操作,而malloc/free 不能。
(1)new :分配内存 + 调用类的构造函数 + 初始化 delete:释放内存 + 调用类的析构函数
(2)malloc:只分配内存,不会进行初始化类成员的工作 free只释放内存,不会调用析构函数
二、new/delete是C++运算符,能重载
(1)new、delete 是运算符,可以进行重载
(2)malloc,free是标准库函数,不可以进行重载,但可以覆盖。
三、new delete 更加安全,简单:
(1)不用计算类型大小:自动计算要分配存储区的字节数
(2)不用强制类型转换:自动返回正确的指针类型(二者返回值不同,一个为void*,一个是某种数据类型指针)
四、new可以分配一个对象或对象数组的存储空间,malloc不可以
五、new和delete搭配使用,malloc和free搭配使用:混搭可能出现不可预料的错误
六、new后执行的三个操作:(某面试题目)
(1)new的类分配内存空间。
(2)调用类的构造方法。
(3)返回该实例(对象)的内存地址
4、构造函数、析构函数与虚函数的关系
(1)为什么构造函数不能是虚函数?
简单点说,构造函数的调用是发生多态的前提(多态有对象指针引发)。
(2)为什么在派生类中的析构函数常常为虚析构函数?
简单点说,发生多态时,防止漏掉调用派生类的析构函数。
(3)把所有的类的析构函数都设置为虚函数好吗?
简单点说,不好,会造成时间和空间浪费。
具体答案在博客构造函数、析构函数与虚函数的关系中。
5、C++中,哪些函数不可以被声明为虚函数
总的来说,共有五种,普通函数(非成员函数)、构造函数、内联函数、静态函数、友元函数。
首先说明两点:
(1)虚函数是为了实现多态,而多态是属于动态联编,在运行时确定调用哪个函数。
(2)虚函数调用时,类之间需要有公有继承 +继承关系 + 基类指针或引用调用。
具体解释
(1)普通函数为啥不能是虚函数?
原因:多态是依托于类的,要声明的多态的函数前提必须是虚函数。
(2)构造函数为啥不能是虚函数?
原因:多态是依托于类的,多态的使用必须是在类创建以后,而构造函数是用来创建构造函数的,所以不行。
具体的原因:虚表指针的初始化时在构造函数进行的,而虚函数需要放到虚表中。在调用虚函数前,必须首先知道虚表指针,此时矛盾就出来了。
(3)内联函数为啥不能是虚函数?
原因:内联函数属于静态联编,即内联函数是在编译期间直接展开,可以减少函数调用的花销,即是编译阶段就确定调用哪个函数了。但是虚函数是属于动态联编,即是在运行时才确定调用哪一个函数。显然这两个是冲突的。
(4)静态函数为啥不能使虚函数?
原因:
<1>从技术层面上说,静态函数的调用不需要传递this指针。但是虚函数的调用需要this指针,来找到虚函数表。相互矛盾
<2>从存在的意义上说,静态函数的存在时为了让所有类共享。可以在对象产生之前执行一些操作。与虚函数的作用不是一路的。
(5)友元函数为啥不能是虚函数?
原因:C++不支持友元函数的继承,不能继承的函数指定不是虚函数。
6、能做switch()的参数类型是:能自动转换为整形(int)且转换过程中不存在精度损失的类型
具体包括:byte,short,char,int. 但是不包括:long,string,double,float等。
7、堆栈溢出一般是由什么原因导致的?
(1)没有回收垃圾资源
(2)递归调用的层次太深
8、C++中的空类,默认产生哪些类成员函数?
(1)默认的构造函数
(2)默认的拷贝构造函数
(3)默认的赋值函数
(4)默认的析构函数
(5)默认的取地址运算符(不带const)
(6)默认的取地址运算符(带const)
class Empty { public: Empty(); // 缺省构造函数 Empty( const Empty& ); // 拷贝构造函数 ~Empty(); // 析构函数 Empty& operator=( const Empty& ); // 赋值运算符 Empty* operator&(); // 取址运算符 const Empty* operator&() const; // 取址运算符 const };
9、面向对象的四个特性:
(1)抽象性:是对事物的抽象概括描述,继而将客观事物抽象成类,从而实现了客观世界向计算机世界的转化。
(2)封装性:把客观事物封装成抽象的类,并且只让可信的类活对象进行访问,而对其他成员进行信息隐藏。
(3)继承性:新产生的类无需重新编写现有类的代码,而直接使用现有类的功能,继而对新类进行扩展。
(4)多态性:允许基类指针指向各种各样的派生类对象,之后能够根据其指向的派生类对象,做出不同的反应(该反应为虚函数实现的)。即多态的出现,就是给出一个统一处理各种各样的派生类的方法。比如可以定义一个数组,数组类型为基类指针类型,放的可以是各种派生类,之后可以使用基类指针对数组元素统一处理。
注:有书上说是四个,有书上说是三个,不是很统一。
10、简述strcpy与memcpy的相同点和区别点:
相同点:strcpy与memcpy都可以实现拷贝的功能
不同点:
(1)实现功能不同,strcpy主要实现字符串变量间的拷贝,memcpy主要是内存块间的拷贝。
(2)操作对象不同,strcpy的操作对象是字符串,memcpy 的操作对象是内存地址,并不限于何种数据类型。
(3)执行效率不同,memcpy最高,strcpy次之。
11、内存分配方式
一个C、C++程序编译时内存分为5大存储区:全局区、栈区、堆区、文字常量区、程序代码区。
(1) 在静态存储区域分配
控制者:编译器
分配时间:在程序编译的时候分配内存
释放时间:在程序的整个运行期间都存在,程序结束后由OS释放
内容:全局变量,static变量
特点:
0、速度快,不易出错。
1、初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和静态变量在另一块区域
2、定义后,变量的值可以改变
(2) 在栈上创建
控制者:由编译器自动分配释放
分配时间:在程序运行(执行函数)的时候分配内存
释放时间:在函数执行结束时,释放存储单元自动被释放。
举例: 局部变量,函数参数
特点:
1、栈内存分配运算 内置于处理器的指令集中,分配效率很高,但是分配的内存容量有限。
2、定义后,变量的值可以改变
(3) 从堆上分配
控制者:程序员一般由程序员分配和释放,。
分配时间:在程序运行(遇见new或malloc)的时候分配内存。
释放时间:程序员自己决定,若程序员不释放,程序结束时可能由OS回收,但是程序运行期间不释放的内存属于内存泄露。
举例:使用new 和 malloc申请的空间
特点:
0、频繁地分配和释放不同大小的堆空间将会产生堆内碎块
1、程序员使用malloc 或new 申请任意多少的内存,自己负责在何时用free 或delete 释放内存,否则会造成内存泄露。
2、定义后,变量的值可以改变
(4) 文字常量区
控制着:编译器
分配时间:在程序编译的时候分配内存
释放时间:程序结束后由系统释放
举例:常量字符串
特点:定义后,变量的值不可以 改变,只读的
(5) 程序代码区
内容:存放函数体的二进制代码
12.堆内存与栈内存的比较
(1)申请方式
栈:由系统分配,
堆:有程序员显式分配,malloc和new
(2)申请大小的限制
栈:在Windows下,栈是向低地址扩展的数据结构,是一块连续的内存的区域。栈内存的地址和大小是系统预先规定好的,而且能从栈申请的大小很小,在WINDOWS下,栈的大小是2M,如果申请的空间超过栈的剩余空间时,将提示overflow。
堆:堆是向高地址扩展的数据结构,是不连续的内存区域。这是由于系统是用链表来存储的空闲内存地址的,自然是不连续的,而链表的遍历方向是由低地址向高地址。堆的大小受限于计算机系统中有效的虚拟内存。由此可见,堆获得的空间比较灵活,也比较大。
(3)申请后系统的响应
栈:只要栈的剩余空间大于所申请空间,系统将为程序提供内存,否则 将报异常提示栈溢出。
堆:首先应该知道操作系统有一个记录空闲内存地址的链表,当系统收到程序的申请时,会遍历该链表,寻找第一个空间大于所申请空间的堆结点,然后将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。由于找到的堆结点的大小不一定正好等于申请的大小,系统会自动的将多余的那部分重新放入空闲链表中。
(4)申请效率的比较
栈:由系统自动分配,速度较快。但程序员是无法控制的。
堆:由new分配的内存,一般速度比较慢,而且容易产生内存碎片,不过用起来最方便。
(5)堆和栈中的存储内容
栈:在函数调用时,第一个进栈的是主函数中后的下一条指令(函数调用语句的 下一条可执行语句)的地址,然后是函数的各个参数,在大多数的C编译器中,参数是由右往左入栈的,然后是函数中的局部变量。注意静态变量是不入栈的。当本次函数调用结束后,局部变量先出栈,然后是参数,最后栈顶指针指向最开始存的地址,也就是主函数中的下一条指令,程序由该点继续运行。
堆:一般是在堆的头部用一个字节存放堆的大小。堆中的具体内容有程序员安排。
(6)存取效率的比较
栈:存取速度快
堆:存取速度慢
13、如何判断程序中堆和栈增长方向?
栈的生长方式是向下的,向着内存地址减小的方向增长。
堆的生长方式是向上的,向着内存地址增加的方向增长。
怎么判断堆和栈的增长方向呢,其实就是依次申请两个变量,判断两个变量的地址大小。
判断栈的增长方式:
#include <iostream> using namespace std; void Func(int a) { int b = 1; cout<<"&a = "<<&a<<endl; //0012FE8C cout<<"&b = "<<&b<<endl; //0012FE7C } int main() { int a = 1; Func(a); system("pause"); return 1; }分析:在函数Func中,变量a和b都是局部变量,都是在栈上申请的,而且肯定是先申请变量a的内存,后申请b的内存。
此时可以直接比较a和b的地址即可。
根据程序的注释可以知道a的地址大于b的地址,且a先于b申请,即在栈中,先申请的内存地址大,后申请的内存地址小。
从而得出,栈内存的增长方式是由高地址到低地址的。
判断堆的增长方式:
#include <iostream> using namespace std; void Func(int* a) { int* b = new int; cout<<"a = "<<a<<endl; //00395B80 cout<<"b = "<<b<<endl; //00395BB0 } int main() { int* a = new int; Func(a); system("pause"); return 1; }分析: 在函数Func中,变量a和b中存放的都是堆内存,而且肯定是先申请变量a的内存,后申请b的内存。
此时可以直接比较a和b的地址即可。
根据程序的注释可以知道a的地址小于b的地址,且a先于b申请,即在堆中,先申请的内存地址小,后申请的内存地址大。
从而得出,堆内存的增长方式是由低地址到高地址的。
13、class 与 struct的区别
C++对struct进行了扩充,使得struct不仅仅包含变量,还可以包含成员函数、能继承、能重载,能实现多态。
在c++中,struct和class的区别主要在于:
(1)默认访问权限不同。
struct:变量和函数默认的访问权限是public,而class默认的访问权限为private。
(2)默认继承的访问权限不同。
struct:继承权限默认为public,而class默认的继承权限为private。
(3)class这个关键字可用于定义模板参数,但关键字struct不可以用于定义模板参数。
14、STL的set用什么实现的?为什么不用hash?
set是由红黑树实现的,红黑树是一个平衡二叉查找树,用来检测某个元素是否在集合中出现。
如果使用hash实现时,需要为不同类型的数据编写哈希函数,而利用红黑树只需要为不同类型重载operator<就可以了。
15、列举面向对象设计的三个基本要素和五种主要设计原则。
(1)三个基本要素:继承、封装、多态
(2)主要设计原则:单一职责原则、里氏代换原则、依赖倒置原则、接口隔离原则、迪米特原则、开放-封闭原则。
16、分析++it和it++的优劣 - 巨人网络2014校招
1、for(iterator it = V.begin(); it != V.end(); ++it) 2、for(iterator it = V.begin(); it != V.end(); it++)分析:
(1)这两个式子的结果肯定是一样的。
(2)俩式子效率不同。++it返回的是对象引用,而it++返回的是临时的对象。由于it是用户自定义类型的,编译器无法对其进行优化,即无法直接不生成临时对象,进而等价于++it,所以每进行一次循环,编译器就会创建且销毁一个无用的临时对象。
(3)对于int i,i++ 和 ++i 在release下,二者效率是等价的,因为编译器对其进行优化了。
17、分析下面代码,回答问题
#include <iostream> using namespace std; class Test { public: void print() { cout<<"a::print null"<<endl; } void set(int v) { m_val = v; cout<<"a::set val = %d"<<endl; } private: int m_val; }; int main() { Test* a = NULL; a->print(); //这个函数的调用结果是什么? a->set(100); //这个函数的调用结果是什么? system("pause"); return 1; }分析:
a->print():输出a::print null
a->set():没有输出,程序直接崩溃。
原因:
(1)a只是一个类Test的指针,程序并未为其生成Test类型的内存。但是a仅仅调用print函数,该函数没有对内存进行读写,而且该函数属于类共享的,所以从编译器的角度上,调用也是没问题的。
(2)a只是一个类Test的指针,程序并未为其生成Test类型的内存,所以m_val是没有内存空间的,因此为其赋值就相当于为null赋值。由于地址为null的位置其实是0x0000,这是一个特殊的内存,只能读不能写,所以当为该内存赋值时,会崩溃。