C++ 栈内存与堆内存小探究
实验方式:尝试以不同方式创建超大号二维数组
测试代码:
#include <iostream>
using namespace std;
const int maxn=1000000;
class C{
int arr[maxn];
};
//int a[maxn]; 全局大数组
//C a; 含大数组的全局对象
int main(){
//int a[maxn]; 局部大数组
//C a; 含大数组的局部对象
//C a=*new C; 通过new创建的含大数组的对象
//C* a=new C; 通过new创建的指向含大数组的对象的指针
cout<<"hello";
return 0;
}
结果:
全局大数组:正常
含大数组的全局对象:正常
局部大数组:段错误
含大数组的局部对象:段错误
通过new创建的含大数组的对象 :段错误
通过new创建的指向含大数组的对象的指针 :正常
分析:C++中可能存在像java一样的堆栈内存机制,对于不同作用域的变量有不同内存管理策略
知识导入:
这篇文章讲的真是太好了:
关于堆栈的讲解(我见过的最经典的)
1、栈区(stack)— 由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈。
2、堆区(heap) — 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表,呵呵。
3、全局区(静态区)(static)—,全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。 - 程序结束后有系统释放
4、文字常量区—常量字符串就是放在这里的。 程序结束后由系统释放
5、程序代码区—存放函数体的二进制代码。
在WINDOWS下,栈的大小是2M或4M
变量类型+变量名出来的变量都在栈里,new出来的内存都在堆里,全局变量有自己的地儿。栈内存有限,少在栈上开大数组。
达成成就:
Stack Overflow
原写于2019年05月31日 21:38:43
2020.7.13更新
过去这一年我学了汇编语言、操作系统,又读完了《CSAPP》,对内存布局又有了新的理解。对于一个程序来说,它的内存布局可以概括为一张图:
逐个解释一下:
- 内核内存:操作系统内核的代码和数据,无法直接访问这块内存。注意使用各种操作系统提供的函数直接不是访问操作系统的代码,而是通过“中断机制”将控制权暂时交给了操作系统,再由操作系统执行对应的例程,执行完毕后再将控制权交还给用户程序。
- 用户栈:存放了函数内定义的临时变量,传给函数的参数,以及函数的调用栈。所谓的函数的调用栈,上个图就是:
在这幅图里main调用了solve调用了dfs,这一层层嵌套的信息就储存在用户栈中。我们常讲的栈溢出、爆栈、stack over flow指的就是临时变量占内存太多或函数嵌套调用过多,同时用户栈的内存空间又太小,导致了内存溢出。
- 共享库的内存映射区域:我们以快速排序函数qsort()为例,如果每个要用qsort的用户程序都在自己的代码里嵌入qsort的代码就造成了重复,是一种浪费行为。为了节省空间操作系统只在内存中存放一份qsort的代码,对于每个需要调用qsort的程序,都在他们的内存布局里添加一个对应的“映射”,类似于只添加了一个快捷方式,并不占用实际空间。
- 运行时堆:所有通过malloc()函数、new关键字等等方式动态分配的内存都在这一区域。
- 读写段:包括了可读可写的全局变量和用static定义的变量,其中.data指初始化完成的,.bss指初始化未完成的
- 只读代码段:.init指操作系统初始化程序时用到的一段小函数,.text指程序的代码,只能读不能修改,.rodata指程序中定义的各种常量,典型的如提前定义好的字符串常量。
引申阅读:内存中的堆和栈到底是什么 - 简书
参考内容:《深入理解计算机系统 第三版》链接器相关章节
C++ 栈内存与堆内存小探究的更多相关文章
- JavaScript 数据结构与算法之美 - 栈内存与堆内存 、浅拷贝与深拷贝
前言 想写好前端,先练好内功. 栈内存与堆内存 .浅拷贝与深拷贝,可以说是前端程序员的内功,要知其然,知其所以然. 笔者写的 JavaScript 数据结构与算法之美 系列用的语言是 JavaScri ...
- java中栈内存与堆内存(JVM内存模型)
java中栈内存与堆内存(JVM内存模型) Java中堆内存和栈内存详解1 和 Java中堆内存和栈内存详解2 都粗略讲解了栈内存和堆内存的区别,以及代码中哪些变量存储在堆中.哪些存储在栈中.内存中的 ...
- js的栈内存和堆内存
栈内存和堆内存在了解一门语言底层数据结构上,挺重要的,做了个总结 JS中的栈内存堆内存 JS的内存空间分为栈(stack).堆(heap).池(一般也会归类为栈中). 其中栈存放变量,堆存放复杂对象, ...
- JavaScript变量——栈内存or堆内存
原文 http://blog.csdn.net/xdd19910505/article/details/41900693 堆和栈这两个字我们已经接触多很多次,那么具体是什么存在栈中什么存在堆中呢?就 ...
- 牛客网Java刷题知识点之内存的划分(寄存器、本地方法区、方法区、栈内存和堆内存)
不多说,直接上干货! 其中 1)程序计数器:用于指示当前线程所执行的字节码执行到了第几行,可以理解为当前线程的行号指示器.每个计数器志勇赖记录一个线程的行号,所以它是线程私有的. ...
- JVM存储位置分配——java中局部变量、实例变量和静态变量在方法区、栈内存、堆内存中的分配
Java中的变量根据不同的标准可以分为两类,以其引用的数据类型的不同来划分可分为“原始数据类型变量和引用数据类型变量”,以其作用范围的不同来区分可分为“局部变量,实例变量和静态变量”. 根据“Java ...
- JS栈内存与堆内存
㈠JavaScript变量 ⒈分类 ⑴JavaScript中的变量分为基本类型和引用类型. ⑵基本类型就是保存在栈内存中的简单数据段. ⑶引用类型指的是那些保存在堆内存中的对象. ⒉基本类型 基本类 ...
- Java中局部变量、实例变量和静态变量在方法区、栈内存、堆内存中的分配
转自:https://blog.csdn.net/leunging/article/details/80599282 感谢CSDN博主「leunging」的总结分享 ———————————————— ...
- (转)java内存分配分析/栈内存、堆内存
转自(http://blog.csdn.net/qh_java/article/details/9084091) java内存分配分析/栈内存.堆内存 java内存分配分析 本文将由浅入深详细介绍Ja ...
- 目录_Java内存分配(直接内存、堆内存、Unsafel类、内存映射文件)
1.Java直接内存与堆内存-MarchOn 2.Java内存映射文件-MarchOn 3.Java Unsafe的使用-MarchOn 简单总结: 1.内存映射文件 读文件时候一般要两次复制:从磁盘 ...
随机推荐
- SpringCloud和Dubbo?
SpringCloud和Dubbo都是现在主流的微服务架构SpringCloud是Apache旗下的Spring体系下的微服务解决方案Dubbo是阿里系的分布式服务治理框架从技术维度上,其实Sprin ...
- 什么是线程池(thread pool)?
在面向对象编程中,创建和销毁对象是很费时间的,因为创建一个对象要获取内 存资源或者其它更多资源.在 Java 中更是如此,虚拟机将试图跟踪每一个对象, 以便能够在对象销毁后进行垃圾回收.所以提高服务程 ...
- Replicated State Machine和WAL
在阅读raft论文的时候,考虑两个问题: 为什么要用Replicated State Machine?没有其他方式吗 为什么要先写日志再应用到Replicated State Machine,直接应用 ...
- spring 自动装配 bean 有哪些方式?
Spring容器负责创建应用程序中的bean同时通过ID来协调这些对象之间的关系.作为开发人员,我们需要告诉Spring要创建哪些bean并且如何将其装配到一起. spring中bean装配有两种方式 ...
- JavaScript 焦点事件
焦点事件,当一个元素(比如链接或表单)得到或失去焦点时发生. 实例: 1 <!DOCTYPE html> 2 <html lang="en"> 3 < ...
- 学习Apache(二)
反向代理负载均衡之APACHE 一.反向代理1.1 介绍反响代理 反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络上的服务器,并将 ...
- glusterfs架构和原理
分布式存储已经研究很多年,但直到近年来,伴随着谷歌.亚马逊和阿里等互联网公司云计算和大数据应用的兴起,它才大规模应用到工程实践中.如谷歌的分布式文件系统GFS.分布式表格系统google Bigtab ...
- Kali Linux 下安装配置MongoDB数据库 ubuntu 下安装配置MongoDB源码安装数据库
Kali Linux 下安装配置MongoDB数据库 1.下载mongodb.tgz 压缩包: 2.解压到:tar -zxvf mongodb.tgz /usr/local/mongodb 3.创 ...
- 创建TypeScript代码模板(NVS+Yarn+ESLint+Prettier+Husky)
创建TypeScript代码模板(NVS+Yarn+ESLint+Prettier+Husky) Cui, Richard Chikun 本文笔者将带你在Github代码仓库创建TypeScript代 ...
- 使用Node.js版本管理器
使用Node.js版本管理器 完全卸载Node.js 清除Package缓存:npm cache clean --force 卸载Node.js:wmic product where caption= ...