C++面试笔记 - (二)
1、C++从代码到可执行二进制文件的过程
C++和C语言类似,一个C++程序从源码到执行文件,有四个过程,预编译、编译、汇编、链接。
预编译:这个过程主要的处理操作如下:
将所有的#define删除,并且展开所有的宏定义
处理所有的条件预编译指令,如#if、#ifdef
处理#include预编译指令,将被包含的文件插入到该预编译指令的位置。
过滤所有的注释
添加行号和文件名标识。
编译:这个过程主要的处理操作如下:
词法分析:将源代码的字符序列分割成一系列的记号。
语法分析:对记号进行语法分析,产生语法树。
语义分析:判断表达式是否有意义。
代码优化。
目标代码生成:生成汇编代码。
目标代码优化。
汇编:这个过程主要是将汇编代码转变成机器可以执行的指令。
链接:将不同的源文件产生的目标文件进行链接,从而形成一个可以执行的程序。
链接分为静态链接和动态链接。
静态链接:是在链接的时候就已经把要调用的函数或者过程链接到了生成的可执行文件中,就算你在去把静态库删除也不会影响可执行程序的执行;生成的静态链接库,Windows下以.lib为后缀,Linux下以.a为后缀。
动态链接:是在链接的时候没有把调用的函数代码链接进去,而是在执行的过程中,再去找要链接的函数,生成的可执行文件中没有函数代码,只包含函数的重定位信息,所以当你删除动态库时,可执行程序就不能运行。生成的动态链接库,Windows下以.dll为后缀,Linux下以.so为后缀。
2、说说 static****关键字的作用
\1. 定义全局静态变量和局部静态变量:初始化的静态变量会在数据段分配内存,未初始化的静态变量
会在BSS段分配内存。直到程序结束,静态变量始终会维持前值。
\2. 定义静态函数:静态函数只能在本源文件中使用;如 static void func();
\3. 定义静态变量。静态变量只能在本源文件中使用;
\4. 定义类中的静态成员变量:使用静态数据成员,它既可以被当成全局变量那样去存储,但又被隐藏
在类的内部。类中的static静态数据成员拥有一块单独的存储区,而不管创建了多少个该类的对
象。所有这些对象的静态数据成员都共享这一块静态存储空间。
\5. 定义类中的静态成员函数:如静态成员函数也是类的一部分,而不是对象的一部分。所有这些对象的静态数据成员都共享这一块静态存储空间。
此外:当调用一个对象的非静态成员函数时,系统会把该对象的起始地址赋给成员函数的this指针。而静态成员函数不属于任何一个对象,因此C++规定静态成员函数没有this****指针。既然它没有指向某一对象,也就无法对一个对象中的非静态成员进行访问。
3、数组和指针的区别
\1. 概念:
(1)数组:数组是用于储存多个相同类型数据的集合。 数组名是首元素的地址。
(2)指针:指针相当于一个变量,它存放的是其它变量在内存中的地址。 指针名指向了内存的首地址。
\2. 区别:
(1)赋值:同类型指针变量可以相互赋值;数组不行,只能一个一个元素的赋值或拷贝
(2)存储方式:
数组:数组在内存中是连续存放的,数组的存储空间,不是在静态区就是在栈上。
指针:指针很灵活,它可以指向任意类型的数据。指针的类型说明了它所指向地址空间的内存。由于指针本身就是一个变量,再加上它所存放的也是变量,所以指针的存储空间不能确定。
(3)求****sizeof:
数组所占存储空间的内存大小:sizeof(数组名)/sizeof(数据类型)在32位平台下,无论指针的类型是什么,sizeof(指针名)都是4,在64位平台下,无论指针的类型是什么,sizeof(指针名)都是8
4、说说什么是野指针,怎么产生的,如何避免
\1. 概念:野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)
\2. 产生原因:释放内存后指针不及时置空(野指针),依然指向了该内存,那么可能出现非法访问的
错误。这些我们都要注意避免。
\3. 避免办法:
(1)初始化置NULL
(2)申请内存后判空
(3)指针释放后置NULL
(4)使用智能指针
5、说说new和malloc的区别,各自底层实现原理。
\1. new是操作符,而malloc是函数。
\2. new在调用的时候先分配内存,在调用构造函数,释放的时候调用析构函数;而malloc没有构造函数和析构函数。
\3. malloc****需要给定申请内存的大小,返回的指针需要强转;new会调用构造函数,不用指定内存的大小,返回指针不用强转。
\4. new****可以被重载;malloc不行
\5. new****分配内存更直接和安全。
\6. new****发生错误抛出异常,malloc返回null
malloc****底层实现:当开辟的空间小于 128K 时,调用 brk()函数;当开辟的空间大于 128K 时,调用
mmap()。malloc采用的是内存池的管理方式,以减少内存碎片。先申请大块内存作为堆区,然后将堆区分为多个内存块。当用户申请内存时,直接从堆区分配一块合适的空闲快。采用隐式链表将所有空闲块,每一个空闲块记录了一个未分配的、连续的内存地址。new****底层实现:**关键字new在调用构造函数的时候实际上进行了如下的几个步骤:
\1. 创建一个新的对象
\2. 将构造函数的作用域赋值给这个新的对象(因此this指向了这个新的对象)
\3. 执行构造函数中的代码(为这个新对象添加属性)
\4. 返回新对象
6、说说使用指针需要注意什么?
\1. 定义指针时,先初始化为NULL。
\2. 用malloc或new申请内存之后,应该立即检查指针值是否为NULL。防止使用指针值为NULL的内存。
\3. 不要忘记为数组和动态内存赋初值。防止将未被初始化的内存作为右值使用。
\4. 避免数字或指针的下标越界,特别要当心发生“多1”或者“少1”操作
\5. 动态内存的申请与释放必须配对,防止内存泄漏
\6. 用free或delete释放了内存之后,立即将指针设置为****NULL,防止“野指针
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现