Windows 上的内存...非常复杂,询问“我的进程使用了多少内存”实际上是一个无意义的问题。要回答您的问题,让我们先了解一些背景知识。
Windows 上的内存通过ptr = VirtualAlloc(..., MEM_RESERVE, ...) 分配,稍后通过VirtualAlloc(ptr+n, MEM_COMMIT, ...) 提交。
任何保留的内存都只会用完地址空间,因此并不有趣。 Windows 会让你MEM_RESERVE TB 的内存就好了。提交内存确实会消耗资源,但不会像您想象的那样。当您调用提交窗口时,会进行一些总和并且基本上可以计算出来(总物理内存 + 总交换 - 当前提交),如果有足够的空闲空间,您可以分配内存。但是,在您实际使用它之前,Windows 内存管理器实际上不会给您物理内存。
但是,稍后,如果 Windows 的物理 RAM 太紧,它会将您的一些 RAM 交换到磁盘(它可能会压缩它并丢弃未使用的页面,丢弃从文件直接映射的任何内容和其他优化)。这意味着您的程序的总提交和总物理内存使用量可能会有很大不同。这两个数字都很有用,具体取决于您要测量的内容。
还有最后一个重要的警告 - 共享内存。当您加载 DLL 代码时,只读内存 [甚至可能是读/写部分,但这是 COW'd] 可以与其他程序共享。这意味着您的应用需要该内存,但您不能仅针对您的应用计算该内存 - 毕竟它可以共享,因此不会像天真的计数所认为的那样占用那么多的物理内存。
(如果您正在编写游戏或类似游戏,您还需要计算 GPU 内存,但我不是这里的专家)
上述所有优点通常都被应用程序使用的堆包裹起来,而你什么也看不到——你要求并使用内存。并且尽可能地优化。
您可以通过转到详细信息选项卡并查看各种选项来看到这一点 - 提交大小和工作集非常有用。如果您只查看任务管理器中的主窗口并且它只有一个值,我希望您现在明白,所用内存的单个值必须是某种妥协,因为这不是一个有意义的问题。
现在回答你的问题
首先,操作系统确切地知道您的应用保留了多少内存以及它已提交了多少。它不知道的是,您(或更可能是 CRT)正在使用的堆实现是否保留了一些尚未释放回操作系统的已释放内存。堆通常这样做是为了优化 - 从操作系统请求内存并将其释放回操作系统是一项相当昂贵的操作(并且只能在称为页面的大块中完成),因此它们中的大多数都会保留一些。
第二个问题:不要使用该值,转到详细信息并使用那里的值,因为只有你知道你真正想问什么。
编辑:
对于您的评论,是的,但这取决于分配的大小。如果您分配一大块内存(例如 >= 1MB),则 CRT 中的堆通常直接将分配推迟到操作系统,因此释放单个内存实际上会释放它们。对于小的分配,CRT 中的堆向操作系统请求内存页面,然后将其细分以在分配中给出。因此,如果您随后释放所有其他人,您将留下漏洞 - 并且堆不能将这些漏洞还给操作系统,因为操作系统通常只能在整个页面中工作。因此,您在任务管理器中看到的任何内容都会显示所有内存仍在使用中。请记住,此内存不会丢失或泄漏,它只是有效地池化,如果分配要求该大小,它将再次使用。如果你关心这段记忆,你可以使用crt heap statistics famliy of functions 来关注那些——特别是_CrtMemDumpStatistics