【C++八股题整理】内存布局、堆和栈、内存泄露、函数调用栈

2024-09-02 17:36

本文主要是介绍【C++八股题整理】内存布局、堆和栈、内存泄露、函数调用栈,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C++八股题整理

  • 内存布局
    • C++中的内存分配情况
    • 堆和栈的内存有什么区别?
    • 堆内存分配慢如何优化?内存池
    • 内存溢出和内存泄漏是什么?如何避免?
    • 内存碎片是什么?怎么解决?
    • 为什么栈的访问效率比堆高?
    • 函数调用时栈的变化?
    • 函数的参数列表为什么从右往左入栈?

内存布局

C++中的内存分配情况

在这里插入图片描述

区域存储内容分配方式生命周期
(Stack)局部变量、局部常量、函数的参数和返回地址自动分配和释放,由编译器管理。连续的内存块,分配释放速度快函数调用时入栈,返回时释放
(Heap)动态分配的对象和数组newdelete,或mallocfree动态分配和释放。不连续的内存块,易出现碎片,速度慢程序员手动分配和释放
全局/静态存储区全局变量、静态变量程序启动时分配,程序结束时释放与程序的整个生命周期一致
常量区字符串字面量、const修饰的全局和静态变量、虚函数表程序启动时分配与程序的整个生命周期一致
代码区程序的二进制代码由操作系统在程序加载时分配与程序的整个生命周期一致
#include <iostream>int globalVar = 10;  // 全局变量,存储在全局/静态存储区
static int staticGlobalVar = 20;  // 静态全局变量,存储在全局/静态存储区class MyClass {
public:int memberVar;  // 成员变量,存储在对象实例所分配的内存中(栈或堆)static int staticMemberVar;  // 静态成员变量,存储在全局/静态存储区// 构造函数,存储在代码区MyClass(int val) : memberVar(val) {}// 成员函数,存储在代码区void show() {std::cout << "Member Var: " << memberVar << std::endl;}};const int constGlobalVar = 40;  // 全局常量,存储在常量区int main() {int localVar = 50;  // 局部变量,存储在栈中const int constLocalVar = 60;  // 局部常量,存储在栈中(通常编译器优化后会放入寄存器)// 静态创建对象MyClass obj(localVar);  // 对象本身和成员变量存储在栈中// 动态创建对象MyClass* heapObj = new MyClass(localVar);  // 指针heapObj在栈上,对象本身和成员变量存储在堆中// 释放动态分配的对象delete heapObj;  // 释放堆中的内存return 0;
}

堆和栈的内存有什么区别?

特性栈内存堆内存
管理方式自动管理,系统分配和释放手动管理,程序员分配和释放
内存布局连续的内存块,不会产生内存碎片非连续的内存块,会产生内存碎片
分配地址由高向低由低向高
分配速度慢,因为要寻找合适大小的内存块
空间大小较小,通常几MB到几十MB较大,通常可达GB级别
生命周期随函数调用开始和结束程序员控制,直到显式释放
存储内容局部变量、函数调用信息动态分配的对象和数据结构

堆内存分配慢如何优化?内存池

内存池通过一次性预先分配一大块内存,并将其划分为多个固定大小的小块,当需要分配内存时,从这些小块中快速分配,而不是每次调用 malloc;释放内存时,直接将小块返回内存池,而不调用 free。这样大幅减少了频繁的堆内存分配和释放操作的系统开销,降低了内存碎片的风险,显著提升了在高频率、小内存块分配场景下的性能。

#include <iostream>class MemoryAllocator {
private:int poolsize;            // 内存池的总大小int blocksize;           // 每个块的大小int numblocks;           // 内存池中块的数量char* pool;              // 指向内存池的指针bool* used;              // 用于标记每个块是否被使用的布尔数组public:// 构造函数:初始化内存池MemoryAllocator(int ps, int bs) : poolsize(ps), blocksize(bs), numblocks(ps / bs) {pool = new char[poolsize];          // 分配内存池used = new bool[numblocks];         // 分配标记数组for (int i = 0; i < numblocks; i++) {used[i] = false;                // 初始化标记数组,所有块都是未使用的}}// 析构函数:释放分配的内存~MemoryAllocator() {delete[] pool;   // 释放内存池delete[] used;   // 释放标记数组}// 分配内存块void* Alloc() {for (int i = 0; i < numblocks; i++) {if (!used[i]) {                  // 查找第一个未使用的块used[i] = true;             // 标记为已使用return &pool[blocksize * i]; // 返回块的指针}}return nullptr;  // 如果没有可用块,返回nullptr}// 释放内存块void free(void* ptr) {// 检查指针是否在内存池的有效范围内if (ptr >= pool && ptr < pool + poolsize) {// 计算块的索引int index = (static_cast<char*>(ptr) - pool) / blocksize;if (used[index]) used[index] = false; // 标记为未使用}}
};int main() {MemoryAllocator pool(100, 4);  // 创建一个内存池,总大小100字节,每块4字节// 分配两个内存块void* ptr1 = pool.Alloc();std::cout << "Allocated at " << ptr1 << std::endl;void* ptr2 = pool.Alloc();std::cout << "Allocated at " << ptr2 << std::endl;// 释放第二个内存块pool.free(ptr2);std::cout << "Freed at " << ptr2 << std::endl;return 0;
}

内存溢出和内存泄漏是什么?如何避免?

  • 内存泄漏:程序在堆上动态分配内存后,未能正确释放不再需要的内存,导致这部分内存无法被重用,从而使得系统中的可用内存逐渐减少。内存泄漏最终会导致内存溢出。
  • 内存溢出(out of memory,OOM):程序试图分配的内存超过了系统或应用程序所能提供的最大可用内存,导致程序运行失败或异常。内存溢出可能发生在堆或栈上。
    • 堆溢出:通常是因为内存泄漏
      int main() {while (true) { int* ptr = new int[1000000]; } // 不断分配内存,最终会导致堆溢出return 0;
      }
      
    • 栈溢出:通常是因为过深的递归调用
      void recursiveFunction() {int largeArray[10000]; // 大量消耗栈空间recursiveFunction();   // 无限递归调用,最终会导致栈溢出
      }
      int main() {recursiveFunction();return 0;
      }
      

可以使用智能指针来自动管理内存,避免手动管理的复杂性。

内存碎片是什么?怎么解决?

  • 内存碎片:在动态内存管理中,由于频繁的分配和释放内存,导致内存中出现许多大小不一的、无法被利用的空闲块的现象
    • 内部碎片:当分配的内存块比实际需要的内存要大时,未使用的部分称为内部碎片。内部碎片主要出现在固定大小的内存分配策略中
    • 外部碎片:内存中存在足够多的空闲空间总量,但由于它们不连续,无法为大块的内存请求提供服务
  • 解决方法
    • 内存紧缩:通过移动分散的内存块来将合并外部碎片,形成连续的内存块。紧缩通常伴随暂停程序运行,因此在实时系统中不太适用
    • 优化分配策略:最佳适应、最先适应、最后适应、首次适应等
    • 内存池:因为块的大小固定且地址连续,因此能避免外部碎片

为什么栈的访问效率比堆高?

  • 缓存局部性:栈内存分配是线性增长的,符合CPU缓存的局部性原则,访问效率高;而堆内存分配则可能在内存中是离散的,导致缓存命中率低,访问速度相对较慢
  • 线程安全性:每个线程都有自己独立的栈,所以栈上的操作通常是线程安全的,不需要同步机制;堆是全局共享的资源,多个线程访问时可能需要同步机制(如锁),这会进一步降低堆内存分配和访问的效率

函数调用时栈的变化?

参考资料 参考资料
比如main函数调用add(int a , int b)函数:

  1. 从右往左压入add的参数值
  2. 保存call指令的下一条指针的地址
  3. 压入EBP(此时EBP指向main的栈底),EBP=ESP(让EBP指向add的栈底),保存main的栈
  4. 开辟add函数的栈空间
  5. add函数中的指令
  6. add函数保存计算值
  7. ESP=EBP,add函数开始退栈
  8. EBP出栈,就是还原main函数的栈底指针
  9. 清除add函数栈空间,继续执行main

这一过程中开辟出的空间称为栈帧,它包括以下内容:

  1. 被调函数的参数:通常从右向左顺序入栈(根据调用约定),即最后一个参数最先入栈。
  2. 被调函数的返回地址:用于在被调函数执行完毕后,返回到调用函数的下一条指令。
  3. 调用者函数的栈帧指针(ebp):保存调用者的栈底地址,以便在被调函数执行完毕后恢复调用者的栈帧。
  4. 被调函数的局部变量:在栈上为被调函数的局部变量分配的空间。

函数的参数列表为什么从右往左入栈?

主要是为了支持函数的变长参数 示例
但这只是一种约定,有些语言和平台并不这样

这篇关于【C++八股题整理】内存布局、堆和栈、内存泄露、函数调用栈的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1130617

相关文章

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

html5的响应式布局的方法示例详解

《html5的响应式布局的方法示例详解》:本文主要介绍了HTML5中使用媒体查询和Flexbox进行响应式布局的方法,简要介绍了CSSGrid布局的基础知识和如何实现自动换行的网格布局,详细内容请阅读本文,希望能对你有所帮助... 一 使用媒体查询响应式布局        使用的参数@media这是常用的

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Python处理函数调用超时的四种方法

《Python处理函数调用超时的四种方法》在实际开发过程中,我们可能会遇到一些场景,需要对函数的执行时间进行限制,例如,当一个函数执行时间过长时,可能会导致程序卡顿、资源占用过高,因此,在某些情况下,... 目录前言func-timeout1. 安装 func-timeout2. 基本用法自定义进程subp

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

在Spring Boot中浅尝内存泄漏的实战记录

《在SpringBoot中浅尝内存泄漏的实战记录》本文给大家分享在SpringBoot中浅尝内存泄漏的实战记录,结合实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录使用静态集合持有对象引用,阻止GC回收关键点:可执行代码:验证:1,运行程序(启动时添加JVM参数限制堆大小):2,访问 htt

前端CSS Grid 布局示例详解

《前端CSSGrid布局示例详解》CSSGrid是一种二维布局系统,可以同时控制行和列,相比Flex(一维布局),更适合用在整体页面布局或复杂模块结构中,:本文主要介绍前端CSSGri... 目录css Grid 布局详解(通俗易懂版)一、概述二、基础概念三、创建 Grid 容器四、定义网格行和列五、设置行

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve