DDR爱好者之家 Design By 杰米

前言

大家应该都遇到过一种情况,在实际工作中有时需要程序打印出某个进程的内存占用情况以作参考, 下面介绍一种通过Linux下的伪文件系统 /proc 计算某进程内存占用的程序实现方法.下面话不多说了,来一起看看详细的介绍吧。

实现分析

首先, 为什么会有所谓的 伪文件 呢. Linux系统的文件类型大致可分为三类: 普通文件, 目录文件和伪文件. 伪文件不是用来存储数据的, 因此这些文件不占用磁盘空间, 只是存在于内存中. /proc 让你可以与内核内部数据进行交互, 获取有关进程的有用信息.

下面主要介绍一下 /proc 下面的四个文件: /proc/stat , /proc/meminfo , /proc/<pid>/stat , /proc/<pid>/status .

/proc/stat 存放系统的cpu时间, 该文件包含了所有cpu活动的信息.

cpu 72389 2891 16811 1148664 31374 0 67 0 0 0
cpu0 17608 452 3786 288899 6210 0 30 0 0 0
cpu1 18724 926 4598 285844 8911 0 15 0 0 0
cpu2 16803 658 3726 288710 7220 0 7 0 0 0
cpu3 19254 855 4700 285209 9032 0 13 0 0 0
...
...
...

/proc/meminfo 存放系统的内存信息, 通过文件中各个变量的名字便可知其代表的信息.

MemTotal: 4046236 kB
MemFree:  1054440 kB
MemAvailable: 2460060 kB
Buffers:  359688 kB
Cached:  1158056 kB
SwapCached:  0 kB
Active:  2020096 kB
Inactive:  677948 kB
Active(anon): 1181376 kB

...
...
...

/proc/<pid>/stat 存放某个进程的cpu信息

2476 (firefox) S 1773 1910 1910 0 -1 4210688 3413511 1712 757 1 45466 4629 2 7 20 0 57 0 20381 1774743552 150565 18446744073709551615 94844693012480 94844693126372 140732961864784 140732961858304 139747170914269 0 0 4096 33572079 0 0 0 17 2 0 0 1178 0 0 94844695226592 94844695228536 94844713955328 140732961867643 140732961867668 140732961867668 140732961869791 0

/proc/<pid>/status 存放某个进程的cpu信息以及一些综合信息

Name: firefox
State: S (sleeping)
Tgid: 2476
Ngid: 0
Pid: 2476
PPid: 1773
TracerPid: 0
Uid: 1000 1000 1000 1000
Gid: 1000 1000 1000 1000
FDSize: 256
Groups: 4 24 27 30 46 108 124 1000 
NStgid: 2476
NSpid: 2476
NSpgid: 1910
NSsid: 1910
VmPeak: 1722812 kB
VmSize: 1690920 kB
VmLck:   0 kB
VmPin:   0 kB
VmHWM: 684048 kB
VmRSS: 600324 kB
VmData: 993040 kB
VmStk:  192 kB
...
...
...

以上数据都可以通过文件读取的方式来获取. 根据自己实验的需要可以计算相应的数据, 比如 pmem = VmRSS/MemTotal*100 等等.

示例代码

下面只是贴出一个简单的获取某进程当前时刻所占用的实际内存的c代码实现例子.

//get_mem.h
#include <stdlib.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h> 
#include <assert.h>
#define VMRSS_LINE 21//VMRSS所在行, 注:根据不同的系统,位置可能有所区别.
#define pid_t int
int get_phy_mem(const pid_t p)
{
 char file[64] = {0};//文件名
 FILE *fd;   //定义文件指针fd
 char line_buff[256] = {0}; //读取行的缓冲区
 sprintf(file,"/proc/%d/status",p);
 fprintf (stderr, "current pid:%d\n", p);
 fd = fopen (file, "r"); //以R读的方式打开文件再赋给指针fd
 //获取vmrss:实际物理内存占用
 int i;
 char name[32];//存放项目名称
 int vmrss;//存放内存
 //读取VmRSS这一行的数据
 for (i=0;i<VMRSS_LINE-1;i++)
 {
  char* ret = fgets (line_buff, sizeof(line_buff), fd);
 }
 char* ret1 = fgets (line_buff, sizeof(line_buff), fd);
 sscanf (line_buff, "%s %d", name,&vmrss);
 fprintf (stderr, "====%s:%d====\n", name,vmrss);
 fclose(fd);  //关闭文件fd
 return vmrss;
}
int get_rmem(pid_t p)
{
 return get_phy_mem(p);
}
int get_total_mem()
{
 const char* file = "/proc/meminfo";//文件名
 FILE *fd;   //定义文件指针fd
 char line_buff[256] = {0}; //读取行的缓冲区
 fd = fopen (file, "r"); //以R读的方式打开文件再赋给指针fd
 //获取memtotal:总内存占用大小
 int i;
 char name[32];//存放项目名称
 int memtotal;//存放内存峰值大小
 char*ret = fgets (line_buff, sizeof(line_buff), fd);//读取memtotal这一行的数据,memtotal在第1行
 sscanf (line_buff, "%s %d", name,&memtotal);
 fprintf (stderr, "====%s:%d====\n", name,memtotal);
 fclose(fd);  //关闭文件fd
 return memtotal;
}

测试文件:

#include "get_mem.h"
int main()
{
int list[1024];
for(int i = 0; i < 1024; i++)
 list[i] = i;
int mem = get_rmem(getpid());

}

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。

DDR爱好者之家 Design By 杰米
广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
DDR爱好者之家 Design By 杰米

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。