C语言面试高频(内存管理)
1.(内存)堆和栈的区别⭐
堆栈空间分配不同:
- 栈由操作系统自动进行分配和释放,用于存放函数的参数值、局部变量的值等,具有高效性。
- 堆一般由程序员手动进行分配和释放,效率比栈低很多。
堆栈缓存方式不同:
- 栈使用一级缓存,存储在处理器核心中,调用完成后立即释放,速度较快。
- 堆存储在二级缓存或主存中,速度相对较慢。
生长方向:
- 堆:堆的分配方向是向上的,即向地址较大的方向分配。当堆需要扩展时,会向高地址方向增长。
- 栈:栈的分配方向是向下的,即向地址较小的方向分配。当栈需要扩展时,会向低地址方向增长。
生命周期:
- 堆:堆上的内存在分配时并不会被立即释放,需要手动进行内存释放操作。堆上的数据可以在程序的任意位置进行访问,不受函数的调用关系限制。
- 栈:栈上的内存分配和释放是自动进行的,随着函数的调用和返回进行相应的操作。栈上的数据只在特定的作用域内有效,函数执行完成后会自动释放。
空间大小:
- 栈的空间大小一般较小,通常最多为2MB,超过则会报溢出错误。
- 堆的空间比较大,理论上可以接近3GB(对于32位程序来说)。
能否产生碎片:
- 栈操作遵循"后进先出"的原则,不会有内存块从栈中弹出,因此不会产生碎片。
- 堆是通过动态分配内存的方式进行分配和释放,频繁的申请和释放内存可能会引发内存碎片问题。
2.在函数中申请堆内存需要注意什么⭐
- 确保不要错误地返回指向栈内存的指针,因为栈内存会在函数结束时自动释放。
- 避免在函数内部申请临时数组,因为函数执行完成后,该数组会消失。
- 不要返回指向常量区的内存空间,因为它们无法修改且获取它们没有意义。
- 使用传入一级指针无法解决问题,因为函数内部指针的修改不会影响传入的指针。
- 在分配堆内存时,确保空间足够存储所需数据,避免访问越界和产生未定义行为。
解决办法如下:
- 使用二级指针来返回申请的堆内存的地址,通过间接引用来修改指针值,从而确保在函数外部能够获取到堆内存的内容。
- 使用指针函数来解决问题,即返回通过malloc函数申请的堆内存的地址,这样可以在函数外部使用free函数释放该内存。
3.请你说说内存碎片⭐
内存碎片是指在内存管理过程中产生的未被有效利用的零散、不连续的内存空间。主要分为两种类型:内部碎片和外部碎片。
- 内部碎片:是由于固定大小的内存分配方式或对齐要求等原因导致的未被利用的小空间。当分配给进程的内存块大于所需的大小时,其中的剩余空间就成为了内部碎片。
- 外
剩余60%内容,订阅专栏后可继续查看/也可单篇购买
c++/嵌入式面经专栏 文章被收录于专栏
本人2022年毕业于山东大学,目前就职intel。打算把之前校招时做的笔记通过专栏发出来,本专栏适合于C/C++、嵌入式方向就业的同学,本篇面经总结数千篇面经的知识集合,实时更新全网最新的嵌入式/C++最新内容,囊括了C语言、C++、操作系统、计算机网络、嵌入式、算法与数据结构、数据库等一系列知识点,在我看来这些是求职者在面试中必须掌握的知识点。最后呢祝各位能找到自己合适的工作。