是今天和朋友讨论的问题,用free看到的内存在减少,但是看/proc/meminfo 的可用内存却没有减少,但是实际情况是他们的应用跑着跑着内存最后就消耗完了,难搞的是复现需要十来天。

我的内存去哪了?_内存泄漏

我的内存去哪了?_linux_02

meminfo里面的memavailable 属性表示的是可以用的内存大小,它是估计出来的值,并不是一个准确值。

我的内存去哪了?_内存泄漏_03

具体链接如下:

​https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=34e431b0ae398fc54ea​

那如果我想确认某个进程有没有出现内存泄漏,我要怎么确认?

这类问题我一般是遇到的时候才会去分析,进程一般会有自己的进程号,比较简单的办法就是用top -p 进程号判断。

也可以去看/proc/<pid>/status 里面的VMRSS属性

实例:

#include "stdio.h"
#include "stdint.h"
#include "stdlib.h"
#include <unistd.h>
#include <string.h>
#include <stdbool.h>
#define MEMSIZE (1024*10)
int main() {
int *p = NULL;
unsigned long cnt = 0;
while(true) {
p = (int*)malloc(MEMSIZE);
memcpy(p,&cnt,MEMSIZE);
sleep(1);
printf("%ld\n",cnt++);
}
return 0;
}

运行:

gcc memory.c && a.out

查看属性:

我的内存去哪了?_linux_04

free 看到的内存信息是从meminfo里面解析出来的,所以要看系统的具体内存占用,还是要看meminfo,但是meminfo里面有很多很多属性,很少有人能够非常明白的说明清楚。

我尝试说下几个通常见到的

MemTotal

这是内存条大小的容量,但是我们往往看到这个值和我们的内存条大小不同,比如,硬件设计的时候告诉我们使用的内存条大小是2GB,实际上这里可能不是2GB,那是因为系统是开机的时候在bootloader,或者其他部分占用了一部分。

我的内存去哪了?_内存泄漏_05

我的内存去哪了?_内存泄漏_06

MemFree

系统剩余内存大小,MemTotal - MemFree 就是已经使用的内存大小。

MemAvailable

这是一个估算值,并不是精确值。

我们看到的剩余内存MemFree大小,并不是我们系统应用程序可以分配的内存大小,系统中存在的缓存cache/buffer、slab回收内存系统可以从其他地方抠一些内存出来再加上MemFree就等于MemAvailable,应用程序就可以判断这个数值判断自己应该申请的内存空间。

我的内存去哪了?_java_07