实验环境: Win7X64Sp1 + vs2008,  物理内存16GB.

实验结论:

*  进程堆的最大Size并没有使用完剩余的物理内存
    *  每次能分配的最大堆空间接近2M, 不管是私有堆,还是进程堆, 和堆初始Size无关, 
    *  将堆空间用尽后, 累计堆空间总Size接近2000MB, 不管是私有堆,还是进程堆, 和堆初始Size无关,

    1. /// @file       getHeapSize.cpp
    2. /// @brief      尝试得到本进程堆的最大size
    3. #include "stdafx.h"
    4. #include <Windows.h>
    5. #include <tchar.h>
    6. #define MEMORY_REQUESTED_MB (1024 * 1024)       ///< 请求的堆尺寸单位(MB)
    7. size_t  GetSystemPageSize();
    8. bool    GetHeapSize(HANDLE hHeap, size_t & nSizeHeapOnce, size_t & nSizeHeap);
    9. bool    GetHeapAllocTotal(HANDLE hHeap, size_t nSizeHeapOnce, size_t & nSizeHeapTotal); ///< 计算在堆上能分配的空间总和
    10. int _tmain(int argc, _TCHAR* argv[])
    11. {
    12. bool    bRc             =   false;
    13. int     iIndex          =   0;
    14. size_t  nSizeHeap       =   0;
    15. size_t  nSizeHeapOnce   =   0;
    16. size_t  nSizeHeapTotal  =   0;
    17. HANDLE  hHeap           =   GetProcessHeap();
    18. _tprintf(L"when hHeap = GetProcessHeap();\n");
    19. for(iIndex = 0; iIndex < 5; iIndex++)
    20. {
    21. bRc = GetHeapSize(hHeap, nSizeHeapOnce, nSizeHeap);
    22. _tprintf(   L"GetHeapSize() = %s, nSizeHeap = 0x%X, %.3f(MB)\n",
    23. bRc ? L"TRUE" : L"FALSE",
    24. nSizeHeap, 1.0f * nSizeHeap / MEMORY_REQUESTED_MB);
    25. }
    26. /// 对私有堆的测试
    27. nSizeHeap = MEMORY_REQUESTED_MB * 10;
    28. hHeap = HeapCreate(0, GetSystemPageSize() * 4, nSizeHeap);  ///< 建立10MB的私有堆, 每次提交4Page
    29. if(NULL == hHeap)
    30. _tprintf(L"HeapCreate error code = 0x%x\n", GetLastError());
    31. else
    32. {
    33. _tprintf(L"HeapCreate Ok, nSizeHeap = %dMB\n", nSizeHeap / MEMORY_REQUESTED_MB);
    34. for(iIndex = 0; iIndex < 5; iIndex++)
    35. {
    36. bRc = GetHeapSize(hHeap, nSizeHeapOnce, nSizeHeap);
    37. _tprintf(   L"GetHeapSize() = %s, nSizeHeap = 0x%X, %.3f(MB)\n",
    38. bRc ? L"TRUE" : L"FALSE",
    39. nSizeHeap, 1.0f * nSizeHeap / MEMORY_REQUESTED_MB);
    40. }
    41. HeapDestroy(hHeap);
    42. }
    43. /// 测试从私有堆上能分配的最大空间, 依次分配一块, 计算能分配的空间总和
    44. nSizeHeap = MEMORY_REQUESTED_MB * 10;
    45. if(0 == nSizeHeapOnce)
    46. nSizeHeapOnce = GetSystemPageSize();
    47. _tprintf(L"nSizeHeapOnce = 0x%x\n", nSizeHeapOnce);
    48. hHeap = HeapCreate(0, nSizeHeapOnce, nSizeHeap);    ///< 建立10MB的私有堆, 每次提交4Page
    49. if(NULL == hHeap)
    50. _tprintf(L"HeapCreate error code = 0x%x\n", GetLastError());
    51. else
    52. {
    53. _tprintf(L"HeapCreate Ok, nSizeHeap = %dMB\n", nSizeHeap / MEMORY_REQUESTED_MB);
    54. bRc = GetHeapAllocTotal(hHeap, nSizeHeapOnce, nSizeHeapTotal);
    55. _tprintf(   L"GetHeapAllocTotal() = %s, nSizeHeapOnce = 0x%X, %.3f(MB), nSizeHeapTotal = 0x%X, %.3f(MB)\n",
    56. bRc ? L"TRUE" : L"FALSE",
    57. nSizeHeapOnce, 1.0f * nSizeHeapOnce / MEMORY_REQUESTED_MB,
    58. nSizeHeapTotal, 1.0f * nSizeHeapTotal / MEMORY_REQUESTED_MB);
    59. HeapDestroy(hHeap);
    60. }
    61. /** operating results
    62. 将程序编译成Release + win32, 在IDE外直接运行
    63. when hHeap = GetProcessHeap();
    64. GetHeapSize() = TRUE, nSizeHeap = 0x70CBF000, 1804.746(MB)
    65. GetHeapSize() = TRUE, nSizeHeap = 0x70CBF000, 1804.746(MB)
    66. GetHeapSize() = TRUE, nSizeHeap = 0x70CBF000, 1804.746(MB)
    67. GetHeapSize() = TRUE, nSizeHeap = 0x70CBF000, 1804.746(MB)
    68. GetHeapSize() = TRUE, nSizeHeap = 0x70CBF000, 1804.746(MB)
    69. HeapCreate Ok, nSizeHeap = 10MB
    70. GetHeapSize() = TRUE, nSizeHeap = 0x1FF000, 1.996(MB)
    71. GetHeapSize() = TRUE, nSizeHeap = 0x1FF000, 1.996(MB)
    72. GetHeapSize() = TRUE, nSizeHeap = 0x1FF000, 1.996(MB)
    73. GetHeapSize() = TRUE, nSizeHeap = 0x1FF000, 1.996(MB)
    74. GetHeapSize() = TRUE, nSizeHeap = 0x1FF000, 1.996(MB)
    75. nSizeHeapOnce = 0x1ff000
    76. HeapCreate Ok, nSizeHeap = 10MB
    77. GetHeapAllocTotal() = TRUE, nSizeHeapOnce = 0x1FF000, 1.996(MB), nSizeHeapTotal
    78. = 0x7C41C000, 1988.109(MB)
    79. 剩余物理内存是8655M, 看来进程堆的最大Size并没有使用完剩余的物理内存
    80. 经过多次实验, 进程堆的Size大约是1800上下几MB
    81. 建立10MB的私有堆, 测试出的私有堆大小为2MB左右??
    82. 建立了20MB的私有堆, 测试出的私有堆大小也为2MB左右.
    83. 建立了100MB的私有堆, 测试出的私有堆大小也为2MB左右.
    84. 结论: 进程私有堆是程序编译时指定的, 指定多大都没用
    85. 在vs2008中设置对大小为10MB, 10485760
    86. 设置步骤: vs2008工程属性 >> Configuration Properties >> Linker >> System >> Heap Reserver Size
    87. 实验结果:
    88. * 每次能分配的堆空间一直2M左右, 不管是私有堆,还是进程堆, 和堆初始Size无关,
    89. * 能分配的堆空间总和接近2000MB, 不管是私有堆,还是进程堆, 和堆初始Size无关,
    90. */
    91. getchar();
    92. return 0;
    93. }
    94. bool    GetHeapAllocTotal(HANDLE hHeap, size_t nSizeHeapOnce, size_t & nSizeHeapTotal)
    95. {
    96. bool        bRc             =   false;
    97. BYTE *      pBufHeap        =   NULL;
    98. nSizeHeapTotal = 0;
    99. do
    100. {
    101. pBufHeap = static_cast<BYTE *>(HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, nSizeHeapOnce));
    102. if(NULL != pBufHeap)
    103. {
    104. bRc = true;
    105. nSizeHeapTotal += nSizeHeapOnce;
    106. }
    107. } while (NULL != pBufHeap);
    108. return bRc;
    109. }
    110. bool    GetHeapSize(HANDLE hHeap, size_t & nSizeHeapOnce, size_t & nSizeHeap)
    111. {
    112. bool        bRc             =   false;
    113. size_t      nSystemPageSize =   GetSystemPageSize();
    114. int         iMultiple       =   0;          ///< 内存申请容量是SYSTEM_INFO.PageSize的倍数
    115. BYTE *      pBufHeap        =   NULL;
    116. iMultiple = static_cast<size_t>(MEMORY_REQUESTED_MB) / nSystemPageSize;
    117. nSizeHeap = iMultiple * nSystemPageSize;    ///< 从1MB开始尝试
    118. do
    119. {
    120. pBufHeap = static_cast<BYTE *>(HeapAlloc(hHeap, HEAP_ZERO_MEMORY, nSizeHeap));
    121. if(NULL != pBufHeap)
    122. {
    123. if(FALSE == HeapFree(GetProcessHeap(), 0, pBufHeap))
    124. return false;
    125. }
    126. iMultiple *= 2; ///< 快速靠近堆上限制
    127. nSizeHeap = iMultiple * nSystemPageSize;
    128. } while (NULL != pBufHeap); ///< 一直尝试到失败为止
    129. /// 从失败的最大Size往小申请, 第一次成功的Size就是进程堆的Size
    130. nSizeHeap = --iMultiple * nSystemPageSize;
    131. do
    132. {
    133. pBufHeap = static_cast<BYTE *>(HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, nSizeHeap));
    134. if(NULL != pBufHeap)
    135. {
    136. if(FALSE == HeapFree(GetProcessHeap(), 0, pBufHeap))
    137. {
    138. return false;
    139. }
    140. bRc = true;
    141. nSizeHeapOnce = nSizeHeap;
    142. break;
    143. }
    144. nSizeHeap = --iMultiple * nSystemPageSize;
    145. } while (NULL == pBufHeap); ///< 一直尝试到成功为止
    146. return bRc;
    147. }
    148. size_t  GetSystemPageSize()
    149. {
    150. /// SYSTEM_INFO.dwPageSize : 页面大小和页保护和承诺的粒度, 是VirtualAlloc的入参
    151. SYSTEM_INFO sSysInfo;
    152. GetSystemInfo(&sSysInfo);
    153. return sSysInfo.dwPageSize;
    154. }

http://blog.csdn.net/lostspeed/article/details/8423215

experiment : 在私有堆和默认进程堆中, 测试能分配的堆空间总和, 每次能分配的最大堆空间的更多相关文章

  1. 栈 堆 stack heap 堆内存 栈内存 内存分配中的堆和栈 掌握堆内存的权柄就是返回的指针 栈是面向线程的而堆是面向进程的。 new/delete and malloc/ free 指针与内存模型

    小结: 1.栈内存 为什么快? Due to this nature, the process of storing and retrieving data from the stack is ver ...

  2. (适合入门)JVM堆内存相关的启动参数:年轻一代、岁和永久代内存分配

    假设你要观察JVM进程消耗的堆内存,通过命令工具jmap或可视化工具jvisualvm.exe.JVM这些参数的默认启动值.假设你想知道JVM内存分配策略,最开始手动设置这些参数.通过JDK统计结果, ...

  3. Java堆和栈的区别和介绍,JVM的堆和栈

    一.Java的堆内存和栈内存 Java把内存划分成两种:一种是堆内存,一种是栈内存.   堆:主要用于存储实例化的对象,数组.由JVM动态分配内存空间.一个JVM只有一个堆内存,线程是可以共享数据的. ...

  4. JVM内存结构之堆、栈、方法区以及直接内存、堆和栈区别

    JVM内存结构之堆.栈.方法区以及直接内存.堆和栈区别 一.  理解JVM中堆与栈以及方法区 堆(heap):FIFO(队列优先,先进先出):二级缓存:*JVM中只有一个堆区被所有线程所共享:对象和数 ...

  5. new 的对象如何不分配在堆而分配在栈上(方法逃逸等)

    当能够明确对象不会发生逃逸时,就可以对这个对象做一个优化,不将其分配到堆上,而是直接分配到栈上,这样在方法结束时,这个对象就会随着方法的出栈而销毁,这样就可以减少垃圾回收的压力. 如方法逃逸. 逃逸分 ...

  6. iOS程序中的内存分配 栈区堆区全局区

    在计算机系统中,运行的应用程序的数据都是保存在内存中的,不同类型的数据,保存的内存区域不同.一.内存分区 栈区(stack) 由编译器自动分配并释放,存放函数的参数值,局部变量等.栈是系统数据结构,对 ...

  7. iOS开发中的内存分配(堆和栈)

    进程的内存分区 所有进程(执行的程序)都必须占用一定数量的内存,它或是用来存放从磁盘载入的程序代码,或是存放取自用户输入的数据等等.不过进程对这些内存的管理方式因内存用途不一而不尽相同,有些内存是事先 ...

  8. (转)在.NET程序运行过程中,什么是堆,什么是栈?什么情况下会在堆(栈)上分配数据?它们有性能上的区别吗?“结构”对象可能分配在堆上吗?什么情况下会发生,有什么需要注意的吗?

    转自:http://www.cnblogs.com/xiaoyao2011/archive/2011/09/09/2172427.html 在.NET程序运行过程中,什么是堆,什么是栈? 堆也就是托管 ...

  9. NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

随机推荐

  1. [D3] Add hovercard

    The way to add hovercard is Append a div with class 'hovercard' in the tick function, positioning th ...

  2. Log4j 2使用教程 分类: B1_JAVA 2014-07-01 12:26 314人阅读 评论(0) 收藏

    转载自 Blog of 天外的星星: http://www.cnblogs.com/leo-lsw/p/log4j2tutorial.html Log4j 2的好处就不和大家说了,如果你搜了2,说明你 ...

  3. 算法中的优化问题(optimization problem)

    和多数算法不同的是,有些问题的答案不只一个,而是需要在多个答案中,按照一定标准选出"最佳"答案,这类问题就统称为"优化问题"(optimization prob ...

  4. var let 区别

    var a = 5; var b = 10; if (a === 5) { let a = 4; // if 块级作用域 var b = 1; // 函数级作用域 console.log(a); // ...

  5. (转)Oracle EXP-00091解决方法

    转自:http://blog.csdn.net/dracotianlong/article/details/8270136 EXP-: 正在导出有问题的统计信息. . . 正在导出表 WF_GENER ...

  6. 就目前市面上的面试整理来说,最全的BAT大厂面试题整理在这

    原标题:就目前市面上的面试整理来说,最全的BAT大厂面试题整理在这 又到了面试求职高峰期,最近有很多网友都在求大厂面试题.正好我之前电脑里面有这方面的整理,于是就发上来分享给大家. 这些题目是网友去百 ...

  7. Python模块学习笔记— —random

    Python中的random模块用于生成随机数. random.random 函数原型 random.random() 生成一个范围在[0,1)的随机浮点数. import random print ...

  8. Docker基础(一)

    1.安装:安装教程很多,Ubuntu14.04安装比较简单docker[之前使用Ubuntu13.04结果安装了好久也没有安装好,后来就直接是14,04了] 2.docker是容器,那么什么是容器? ...

  9. 七步从AngularJS菜鸟到专家(7):Routing

    这是"AngularJS – 七步从菜鸟到专家"系列的第七篇. 在第一篇,我们展示了如何開始搭建一个AngularaJS应用.在第四.五篇我们讨论了Angular内建的directives.上一篇了解 ...

  10. stm32优先级