文章来源:http://blog.csdn.net/bigbug_zju/article/details/39525281

计算机系统中的堆和栈是跟程序员最密切的两个概念。如果没有栈和堆的概念,下面程序的错误就不知道其所以然。

  1. #include <stdlib.h>
  2. int main(int argc, char* argv[])
  3. {
  4. int a[1024][1024];
  5. system("pause");
  6. return 0;
  7. }

消耗栈空间主要有两种操作:函数调用和局部变量。在写递归程序时,大家就容易碰到栈空间溢出的情况。上述代码就是局部变量申请引发栈溢出错误的例子。在vs2008中测试,具体的错误名称为:stack overflow,即栈发生溢出,也就是说申请的空间a[1024][1024]超出了栈的大小,所以出现栈溢出的错误。

此处我们好奇的一个问题是,栈到底有多少大呢?通过下述的代码,在vs2008下的输出为1011kB;

  1. #include <stdlib.h>
  2. #include <iostream>
  3. #include <windows.h>
  4. using namespace std;
  5. int stackApp(int & count)
  6. {
  7. int a=1;
  8. while(1)
  9. {
  10. _asm{ push a} //消耗栈空间;
  11. count++;
  12. }
  13. }
  14. int _tmain(int argc, _TCHAR* argv[])
  15. {
  16. int count = 0;
  17. __try{
  18. stackApp(count);
  19. }
  20. __except(GetExceptionCode()==STATUS_STACK_OVERFLOW ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH){
  21. cout<<"exception"<<endl;
  22. cout<<"the count is "<<(count*4/1024)<<"kB"<<endl;
  23. }
  24. system("pause");
  25. return 0;
  26. }

读者仔细阅读上述代码的话,可能会觉得有问题。因为调用stackApp也会涉及对栈的操作,消耗栈的空间,但是此处基本上没有影响,一次函数调用消耗的栈空间十分有限(几个字节而已)。经过上述实验,大家对于栈空间的大小,应该有丰满的概念了。

下一个问题就是:如果申请的空间过大,栈不够,那该怎么处理呢?这就要说到堆,我们平常使用的malloc语句就是在堆上进行空间申请,一般申请几百兆空间也不是问题。下面的代码回答了一个问题:在堆上可以最大可申请多少的空间?

  1. #include <stdlib.h>
  2. #include <stdio.h>
  3. #define UNIT (100*1024*1024)
  4. int main(int argc, char* argv[])
  5. {
  6. unsigned long max = 0;
  7. char* p = NULL;
  8. while(1){
  9. max++;
  10. p = (char*)malloc(max*UNIT);
  11. if(p != NULL){   //申请成功输出一个1;
  12. printf("%d ", 1);
  13. free(p);
  14. }else{
  15. printf("frame size: %ld\n", max-1); //输出申请的最大空间;
  16. break;
  17. }
  18. }
  19. getchar();
  20. return 0;
  21. }

上述代码在我的Windows xp,vs2008,内存4G的ThinkPad系统上,最终能够申请的最大空间为1.7G;至于为什么只能申请到这么大空间,可参见《程序员的自我修养》pg309。

---------------------------------------------------------------------------------------

1、栈大小固定(编译时确定),堆的大小实际上(运行时)动态变化的,但有理论最大值:进程空间大小-内核空间大小-栈大小-全局空间。
linux系统下默认栈大小是10M,windows系统下默认栈大小是1M。windows下用vs2010编译C++程序时,编译属性中可以重新设定栈大小.
堆的话,理论上内存有多大,就可以建多大.但32位的程序在64位系统上运行的时候,一个进程的堆大小应该是不可以超过4G的.

2、Windows每个线程的栈都是独立的,一个进程有多少个线程就有多少个栈(问题:啥意思,我一直以为每个函数都有自己的栈)。

3、Win32一个进程一次性能够分配最大的堆空间取决于最大的那个堆。Windows进程地址空间分布有heap0~heap5(头一次听说Windows的堆是有限的)。heap5是最大的堆,大小约1.5GB~1.7GB。

4、VS2010工程属性-链接器-系统-堆栈保留大小,可以修改栈大小。

5、栈:由系统自动分配,速度较快;栈向低地址扩展,内存连续。栈顶的地址和栈的最大容量是系统预先规定好的。

堆:由程序员new分配,速度较慢,容易产生内存碎片。堆向高地址扩展,不连续。堆的大小受限于电脑的虚拟内存。

关于栈和堆的定量分析(★firecat推荐★)的更多相关文章

  1. .NET中六个重要的概念:栈、堆、值类型、引用类型、装箱和拆箱 (转)

    作者: Edison Chou  来源: 博客园  发布时间: 2014-09-03 15:59  阅读: 318 次  推荐: 2   原文链接   [收藏]   原文作者:Shivprasad k ...

  2. java栈内存堆内存和GC相关

    java栈内存堆内存 Java把内存分成两种,一种叫做栈内存,一种叫做堆内存,有着不同的作用.栈内存用来存储局部变量和方法调用.栈内存归属于单个线程,每个线程都会有一个栈内存,其存储的变量只能在其所属 ...

  3. jvm之栈、堆

    1. Java Virtual Machine ​ 人群当中,一位叫java的小伙子正向周围一众人群细数着自己取得的荣耀与辉煌.就在此时,c老头和c++老头缓步走来,看着被众人围住的java,c老头感 ...

  4. 【译】.NET中六个重要的概念:栈、堆、值类型、引用类型、装箱和拆箱

    为何要翻译 一来是为了感受国外优秀技术社区知名博主的高质量文章,二来是为了复习对.NET技术的基础拾遗达到温故知新的效果,最后也是为了锻炼一下自己的英文读写能力.因为是首次翻译英文文章(哎,原谅我这个 ...

  5. .NET中的六个重要概念:栈、堆、值类型、引用类型、装箱和拆箱

    为何要翻译 一来是为了感受国外优秀技术社区知名博主的高质量文章,二来是为了复习对.NET技术的基础拾遗达到温故知新的效果,最后也是为了锻炼一下自己的英文读写能力.因为是首次翻译英文文章(哎,原谅我这个 ...

  6. c#中栈和堆的理解

    之前对栈(stack)和堆(heap)的认识很模糊,今天看了一篇关于堆栈的文章<译文---C#堆VS栈>后,仿佛有种拨开云雾见青天的感觉,当然只是一些浅显的理论的认识,这里做一些简单的记录 ...

  7. 值类型,引用类型,栈,堆,ref,out

    在网上收集... C#的值类型,引用类型,栈,堆,ref,out C# 的类型系统可分为两种类型,一是值类型,一是引用类型,这个每个C#程序员都了解.还有托管堆,栈,ref,out等等概念也是每个C# ...

  8. 关于VS中更改栈和堆空间的大小

    编号:1008时间:2016年4月12日17:01:38功能:关于VS中更改栈和堆空间的大小 URL:http://blog.csdn.net/icerock2000/article/details/ ...

  9. c语言中静态区,栈,堆的理解

    对于程序员,一般来说,我们可以简单的理解为内存分为三个部分:静态区,栈,堆. 很多书没有把把堆和栈解释清楚,导致初学者总是分不清楚. 其实堆栈就是栈,而不是堆. 堆的英文是heap:栈的英文是stac ...

随机推荐

  1. 设计模式(九):Composite组合模式 -- 结构型模式

    1. 概述 在数据结构里面,树结构是很重要,我们可以把树的结构应用到设计模式里面. 例子1:就是多级树形菜单. 例子2:文件和文件夹目录 2.问题 我们可以使用简单的对象组合成复杂的对象,而这个复杂对 ...

  2. HDOJ(HDU) 1406 完数

    Problem Description 完数的定义:如果一个大于1的正整数的所有因子之和等于它的本身,则称这个数是完数,比如6,28都是完数:6=1+2+3:28=1+2+4+7+14. 本题的任务是 ...

  3. 抓取“矢量”的实时交通流量数据

    1. 引言 最近老师有一个需求,就是想要抓取实时的矢量交通流量数据来做分析,类似于百度地图,高德地图的"实时路况"那种.平时的网络抓取工作一般是抓取网页上现成的数据,但是交通流量数 ...

  4. ORACLE_CLASS_ENDING

    [JSU]LJDragon's Oracle course notes In the first semester, junior year Oracle考前复习 试题结构分析: 1.选择题2x10, ...

  5. 关于使用百度ueditor时的一些问题

    本来这些问题直接在百度贴吧里回答不就完事了,可是好死不死的,百度贴吧里老出现 未知错误,错误号:230274 看来还是算了,自己做一个随笔记录一下好了 关于我们获取里面的内容时,老是会有一个<p ...

  6. 将vim改造成C/C++开发环境(IDE) 2011

        [参考资料]吴垠的“手把手教你把Vim改装成一个IDE编程环境”在Fedora下成功将Vim打造成适用于C/C++的IDE用Vim搭建C/C++开发环境 Ubuntu下vim+ctags的配置 ...

  7. 搭建Windows下Java Web开发环境

      概要 1.SSH开发相关软件及开发包下载2.软件安装及相关设置3.最简单的Web程序 1.软件下载 在D盘建一个目录JavaTools,用来存放下载的软件和开发包.(本教程将使用D盘,你也可以使用 ...

  8. Java和C++的不同

    现在一边继续深入C++,一边学习Java,为了学习得更加透彻,不断比较两者之间的不同,以后会慢慢继续增加. 1.在多态的实现上,C++需要利用关键字virtual,而Java不需要,因为在Java中, ...

  9. 笔记本分享无线Wifi

    两种方法: 一.使用软件分享,如Wifi共享精灵,设置非常简单. 二.开启windows 7的隐藏功能:虚拟WiFi和SoftAP(即虚拟无线AP),就可以让电脑变成无线路由器,实现共享上网,节省网费 ...

  10. (转) 将VB.NET网站转换成C#的全过程

    在学习URL重写过程中碰到个是VB写的源码,看起来总是不爽的就GOOLE了下 感觉这个文章写的不错 原文地址 http://www.cnblogs.com/cngunner/archive/2006/ ...