【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. 被调函数的局部变量:在栈上为被调函数的局部变量分配的空间。

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/51879.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

UI自动化测试 —— web端元素获取元素等待实践!

前言 Web UI自动化测试是一种软件测试方法&#xff0c;通过模拟用户行为&#xff0c;自动执行Web界面的各种操作&#xff0c;并验证操作结果是否符合预期&#xff0c;从而提高测试效率和准确性。 目的&#xff1a; 确保Web应用程序的界面在不同环境(如不同浏览器、操作系统)下…

【前缀和算法】--- 进阶题目赏析

Welcome to 9ilks Code World (๑•́ ₃ •̀๑) 个人主页: 9ilk (๑•́ ₃ •̀๑) 文章专栏&#xff1a; 算法Journey 本篇我们来赏析前缀和算法的进阶题目。 &#x1f3e0; 和可被K整除的子数组 &#x1f4cc; 题目解析 和可被k整除的子数组 &#x1f4cc; …

记一次ssh伪终端修改为shell

问题 用户ssh进行连接后&#xff0c;默认为伪终端。 解决办法&#xff0c;可以先拿到终端shell&#xff0c;查看用户是否为/bin/bash&#xff1a; 不是/bin/bash&#xff0c;使用如下命令进行修改&#xff1a; chsh -s /bin/bash rootservice sshd restart

量化投资策略与技术学习PART1.1:量化选股之再谈多因子模型(二)

在上一个多因子模型中&#xff0c;我手动对各个因子进行了回测&#xff0c;但是数据结果并不是十分理想&#xff0c;难道基本面指标真的和股票走势关系不大么&#xff1f; 这里我还是准备再测试一下&#xff0c;策略如下&#xff1a; &#xff08;1&#xff09;首先我获取了一下…

codeforces Round 970 (Div. 3)(A-F)

文章目录 [Codeforces Round 970 (Div. 3)](https://codeforces.com/contest/2008)A-[Sakurakos Exam](https://codeforces.com/contest/2008/problem/A)B-[Square or Not](https://codeforces.com/contest/2008/problem/B)C-[Longest Good Array](https://codeforces.com/cont…

Ubuntu上安装配置(jdk/tomcat/ufw防火墙/mysql)+mysql卸载

jdk安装 1.上传jdk压缩包 详情&#xff1a; 下载rz服务&#xff08;lrzsz&#xff09;&#xff1a;sudo apt install lrzsz(在主用户root就不用sudo)下载压缩包&#xff1a;rz 2.解压jdk压缩包 &#xff1a; 详情&#xff1a; 在压缩包所在位置&#xff08;解压压缩使用看Li…

测试人如何高效地设计自动化测试框架?

关于测试框架的好处&#xff0c;比如快速回归提高测试效率&#xff0c;提高测试覆盖率等这里就不讨论了。这里主要讨论自动化框架包含哪些内容&#xff0c;以及如何去设计一个测试框架。 什么是自动化测试框架&#xff1f; 它是由一个或多个自动化测试基础模块、自动化测试管…

Qt22双缓冲机制

Qt22双缓冲机制 知识点drawwidgetdrawwidget.hdrawwidget.cpp mainwindowmainwindow.hmainwindow.cpp main.cpp运行图 知识点 双缓冲就是在内存区申请一块缓存&#xff1b;然后显卡直接从这块内存读取数据.。 这样就不用鼠标边画&#xff0c;经过IO来读取这个环节&#xff1b;…

EasyExcel实现复杂Excel的导入

最近项目中遇到一个复杂的Excel的导入&#xff0c;并且数据量较大。因为数据不规则&#xff0c;所以只能使用POI进行自定义读取&#xff0c;但是发现数据量大之后&#xff0c;读取数据非常耗时。后面换成EasyExcel&#xff0c;性能起飞。 1. Excel样板 如上图&#xff0c;需要…

【C++】汇编分析,函数是如何调用,传参,返回

传参 有的是用寄存器传参&#xff0c;有的用push传参 我在MSVC编译测出来的是PUSH传参&#xff08;debug模式&#xff09;&#xff0c;具体过程如下 long func(long a, long b, long c, long d,long e, long f, long g, long h) {long sum;sum (a b c d e f g h);ret…

VMware安装windows虚拟机详细过程

目录 准备工作配置虚拟机为虚拟机设置操作系统镜像安装windows10 准备工作 安装好VMware软件并激活&#xff0c;激活码自行查找 准备好系统镜像文件&#xff0c;可以在MSDN中下载&#xff0c;地址&#xff1a;https://next.itellyou.cn/ 配置虚拟机 选择自定义 默认 选择稍后…

骨灵冷火!Solon Cloud Gateway 照面发布

骨灵冷火&#xff0c;是练药的好火哟。极冷&#xff0c;又极热。在冰冻中被烧死&#xff1a;&#xff09; 1、认识 Solon Cloud Gateway Solon Cloud Gateway 是基于 Solon Cloud、Vert.X 和 Solon Rx(reactive-streams) 接口实现。小特点&#xff1a; 纯响应式的接口体验流…

[Linux]:基本指令(上)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;Linux学习 贝蒂的主页&#xff1a;Betty’s blog 与Windows环境不同&#xff0c;我们在linux环境下需要通过指令进行各操作&…

13.DataLoader 的使用

DataLoader 的使用 dataset&#xff1a;告诉程序中数据集的位置&#xff0c;数据集中索引&#xff0c;数据集中有多少数据&#xff08;想象成一叠扑克牌&#xff09;dataloader&#xff1a;加载器&#xff0c;将数据加载到神经网络中&#xff0c;每次从dataset中取数据&#x…

Zynq7000系列FPGA中的DDRC纠错码(ECC)

仅在半总线宽度&#xff08;16位&#xff09;数据宽度配置中提供可选的ECC支持。这种配置下&#xff0c;外部DRAM DDR设备需要26位&#xff0c;其中16位用于数据&#xff0c;10位用于ECC。每个数据字节使用独立的5位ECC字段&#xff0c;这种模式提供了单错误纠正和双错误检测的…

UE5蓝图 抽卡出货概率

SSR概率0.1 SR概率0.2 R概率0.7 ps&#xff1a;数组内相加为1。且从小到大排序。

C练手题--Two Oldest Ages 【7 kyu】

一、原题 链接&#xff1a;Training on Two Oldest Ages | Codewars The two oldest ages function/method needs to be completed. It should take an array of numbers as its argument and return the two highest numbers within the array. The returned value should b…

网络-多路io

了 fcntl 函数来操作文件描述符的状态标志&#xff0c;其中主要是为了设置非阻塞模式。下面是对 fcntl 函数及其参数的详细解释&#xff1a; fcntl 函数 fcntl 是一个用于操作文件描述符的系统调用&#xff0c;可以用来设置或获取文件描述符的各种属性。其原型如下&#xff1…

Ubuntu Linux Server安装Kubernetes

本文主要描述在Ubuntu Linux Server操作系统中安装Kubernetes云原生对应的microk8s组件。 sudo snap install microk8s --classic 如上所示&#xff0c;在Ubuntu服务器中安装microk8s组件完成&#xff0c;对应的版本是microk8s v1.30版本 microk8s enable dashboard 如上所…

华为云征文|基于Flexus云服务器X实例的应用场景-定时给微信群中推送新闻简报

&#x1f534;大家好&#xff0c;我是雄雄&#xff0c;欢迎关注微信公众号&#xff1a;雄雄的小课堂 先看这里 写在前面效果华为云Flexus X实例服务器部署开源的热点新闻项目ssh连接服务器docker部署今日热点项目访问今日热点项目 搭建微信交互工具获取token创建发送的公共方法…