C ++ 中的栈内存和堆内存的区别

本文详细解析了C++中栈内存和堆内存的特点及应用,对比了数据结构中的堆与栈,深入探讨了内存分配机制,包括自动分配的栈内存与程序员手动管理的堆内存,以及它们在程序运行中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

C ++ 中的栈内存和堆内存的区别

  • 数据结构中的堆与栈

栈:是一种连续储存的数据结构,具有先进后出的性质。通常的操作有入栈(压栈)、出栈和栈顶元素。想要读取栈中的某个元素,就要将其之前的所有元素出栈才能完成。类比现实中的箱子一样。

堆:是一种非连续的树形储存数据结构,每个节点有一个值,整棵树是经过排序的。特点是节点的值最小(或最大),且根节点的连个子树也是一个堆。常用来实现优先队列,存取随意。

  • 内存中的栈区与堆区:

一般说到内存,指的是计算机的随机储存器(RAM),程序都在这里面运行。计算机内存的大致划分如下图所示:

栈内存:由程序自动向操作系统申请分配以及回收,速度快,使用方便,但程序员无法控制。若分配失败,则提示栈溢出错误。注意:const局部变量也存储在栈区内,栈区向地址减小的方向增长。(在VS2017中是增加的)

测试代码:

//测试栈内存
#include <iostream>
using namespace std;

int main()
{
	int i = 1;
	const int j = 2;
	int k = 3;
	cout << "the address of i is " << &i << endl;
	cout << "the address of j is " << &j << endl;
	cout << "the address of k is " << &k << endl;

	system("pause");
	return 0;
}

VS2017结果图:

&i < &j < &k,地址增加

Sublime Text3结果图:

&i > &j > &k,地址减小

堆内存:程序员向操作系统申请一块内存,当系统收到程序的申请时,会遍历一个记录空闲内存地址的链表,寻找第一个空间大于所申请空间的堆结点,然后将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。分配的速度较慢,地址不连续,容易碎片化。此外,由程序员申请,同时也必须由程序员负责销毁,否则将导致内存泄漏。

//测试堆内存和栈内存
#include <iostream>
using namespace std;

int main()
{
	int i = 1;					//存储在栈区中
	char pc[] = "hello world!";			//存储在栈区中
	const double cd = 3.14;				//存储在栈区中
	static long si = 77;				//存储在可读写区,专门用来存储全局变量和静态变量的内存
	int* pi = new int(100);				//存储在堆区中,运行时才分配内存

	cout << "the address of i is " << &i << endl;
	cout << "the address of pc is " << &pc << endl;
	cout << "the address of cd is " << &cd << endl;
	cout << "the address of si is " << &si << endl;
	cout << "the address of pi is " << pi << endl;

	delete pi;					//释放内存
	system("pause");
	return 0;
}


VS2017结果图:

Sublime Text 3结果图:

  • 关于堆和栈区别的比喻

使用栈就像我们去饭馆里吃饭,只管点菜(发出申请)、付钱、和吃(使用),吃饱了就走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处就是快捷,但是自由度小。
使用堆就像是自己动手做 喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由度大。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值