近期在公司中解决程序使用的内存高问题,将一部分之前无法回收的内存进行了回收,实现降内存效果(降实存)。
在统计效果时, qa问是统计res(实存)还是virt(虚存)。
在网上学习看了一些博客,这里自己总结一下res和virt的区别。
1. 概念
virt:
1、进程“需要的”虚拟内存大小,包括进程使用的库、代码、数据,以及malloc、new分配的堆空间和分配的栈空间等; 2、假如进程新申请10mb的内存,但实际只使用了1mb,那么它会增长10mb,而不是实际的1mb使用量。 3、virt = swap res
res:
1、进程当前使用的内存大小,包括使用中的malloc、new分配的堆空间和分配的栈空间,但不包括swap out量; 2、包含其他进程的共享; 3、如果申请10mb的内存,实际使用1mb,它只增长1mb,与virt相反; 4、关于库占用内存的情况,它只统计加载的库文件所占内存大小。 5、res = code data
shr:
1、除了自身进程的共享内存,也包括其他进程的共享内存; 2、虽然进程只使用了几个共享库的函数,但它包含了整个共享库的大小; 3、计算某个进程所占的物理内存大小公式:res – shr; 4、swap out后,它将会降下来。
2. 测试
(1) 使用堆分配内存, 进行测试:
1 #include2 #include 3 #include 4 5 int main() 6 { 7 int test = 0; 8 //分配512m, 未使用 9 char * p = new char [1024*1024*512]; 10 scanf("%d", &test); //等待输入 11 12 //使用10m 13 memset(p, 0, 1024 * 1024 * 10); 14 scanf("%d", &test); //等待输入 15 16 //使用50m 17 memset(p, 0, 1024 * 1024 * 50); 18 scanf("%d", &test); //等待输入 19 delete [] p; 20 return 0; 21 }
执行:
new 512m后,virt/res
使用10m后,virt/res
使用50m后,virt/res
(2) 使用栈分配内存进行测试:
1 #include2 #include 3 #include 4 5 int main() 6 { 7 int test = 0; 8 //20m栈, 未使用 9 char p[1024*1024*20]; 10 scanf("%d", &test); //等待输入 11 12 //使用10m 13 memset(p, 0, 1024 * 1024 * 10); 14 scanf("%d", &test); //等待输入 15 return 0; 16 }
执行:
栈上申请20m
使用10m
3. 总结
堆、栈分配的内存,如果没有使用是不会占用实存的,只会记录到虚存。
如果程序占用实存比较多,说明程序申请内存多,实际使用的空间也多。
如果程序占用虚存比较多,说明程序申请来很多空间,但是没有使用。
工作中,遇到过有的程序虚存300g , 实存只有不到15g。