Linux系統下輸出某進程內存占用信息的c程序實現


在實際工作中有時需要程序打印出某個進程的內存占用情況以作參考, 下面介紹一種通過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());

}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM