分享:堆和栈的区别
堆和栈的区别
一、预备知识-程序的内存分配
一个由C/C++编译的程序占用的内存分为以下几个部分
1、栈区(stack)- 由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其
操作方式类似于数据结构中的栈。
2、堆区(heap) - 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回
收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表,呵呵。
3、全局区(静态区)(static)-,全局变量和静态变量的存储是放在一块的,初始化的
全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另
一块区域。 - 程序结束后由系统释放。
4、文字常量区 -常量字符串就是放在这里的。 程序结束后由系统释放
5、程序代码区-存放函数体的二进制代码。
总结:
堆和栈的区别可以用如下的比喻来看出:
使用栈就象我们去饭馆里吃饭,只管点菜(发出申请)、付钱、和吃(使用),吃饱了就
走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处是快捷,但是自
由度小。
使用堆就象是自己动手做喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由
度大。
[解决办法]
顶 sf
[解决办法]
讲的不错.
[解决办法]
收藏了。。。。
[解决办法]
楼主这么说的话很容易误导菜鸟啊.
[解决办法]
直接在进程的地址空间中保留一块内存,虽然用起来最不方便。但是速度快,也最灵活。
2.5堆和栈中的存储内容
栈: 在函数调用时,第一个进栈的是主函数中后的下一条指令(函数调用语句的下一条可
执行语句)的地址,然后是函数的各个参数,在大多数的C编译器中,参数是由右往左入栈
的,然后是函数中的局部变量。注意静态变量是不入栈的。
当本次函数调用结束后,局部变量先出栈,然后是参数,最后栈顶指针指向最开始存的地
址,也就是主函数中的下一条指令,程序由该点继续运行。
堆:一般是在堆的头部用一个字节存放堆的大小。堆中的具体内容由程序员安排。
2.6存取效率的比较
char s1[] = "aaaaaaaaaaaaaaa";
char *s2 = "bbbbbbbbbbbbbbbbb";
aaaaaaaaaaa是在运行时刻赋值的;
而bbbbbbbbbbb是在编译时就确定的;
但是,在以后的存取中,在栈上的数组比指针所指向的字符串(例如堆)快。
比如:
#include
void main()
{
char a = 1;
char c[] = "1234567890";
char *p ="1234567890";
a = c[1];
a = p[1];
return;
}
对应的汇编代码
10: a = c[1];
00401067 8A 4D F1 mov cl,byte ptr [ebp-0Fh]
0040106A 88 4D FC mov byte ptr [ebp-4],cl
11: a = p[1];
0040106D 8B 55 EC mov edx,dword ptr [ebp-14h]
00401070 8A 42 01 mov al,byte ptr [edx+1]
00401073 88 45 FC mov byte ptr [ebp-4],al
第一种在读取时直接就把字符串中的元素读到寄存器cl中,而第二种则要先把指针值读到
edx中,再根据edx读取字符,显然慢了。
2.7小结:
堆和栈的区别可以用如下的比喻来看出:
使用栈就象我们去饭馆里吃饭,只管点菜(发出申请)、付钱、和吃(使用),吃饱了就
走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处是快捷,但是自
由度小。
使用堆就象是自己动手做喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由
度大。
[解决办法]
在c#中
一般来说声明的对象(值类型) 将会存放在栈中 对于引用此对象的方法或者类将得到栈中对象的地址.
如果栈中存放的是对象数组 则存放了数组在堆中的首地址.
如果栈中存放的对象是指针对象 则与对象数组一样存放了指针指向的地点的首地址.
对于引用了变量的方法或者类将得到对象在栈中的首地址,
通过首地址得到栈中结果,通过接过得到堆中数据(值类型将会直接存储在栈中而不会存放到堆中)
[解决办法]
恩,好东西
[解决办法]
笑而不语。。。
[解决办法]
程序员可分配的是堆,系统分配的是栈。差不多可以这么理解。
[解决办法]
很不赖!
[解决办法]
很好,学习了
[解决办法]
呵呵。讲的一般啊。
[解决办法]
是 不错
[解决办法]
谢谢分享
[解决办法]
学习了。顶!!!
[解决办法]
引用类型 值类型
[解决办法]