Linux 内存占用分析

23 篇文章 0 订阅

这篇博客主要介绍 linux 环境下,查看内存占用的两种方式:使用 ps,top等命令;查看/proc/[pid]/下的文件。文章简要介绍了命令的使用方法与一些参数意义,同时对/proc/[pid]/下的文件内容进行了一些详细的介绍。文章内容来自google和自我总结,如有不当之处,欢迎批评指正。

查看 linux 中内存占用的方法

linux 下面查看内存有多种渠道,比如通过命令 ps ,top,free, pmap 等,或者通过/proc系统。一般情况下,ps,top,pmap,free可以满足要求,如果需要比较详细和精确地知道整机内存或者某个进程内存的使用情况,可以通过/proc 系统。

使用命令

free : 显示系统可用内存以及已经使用的内存的信息
ps: 查看进程信息,静态,即当前状态
top: 查看进程信息,动态
pstree: 查看进程树
pmap: 根据进程ID查看进程信息

ps vs top

  1. ps命令–提供系统过去信息的一次性快照,也就是说ps命令能够查看刚刚系统的进程信息。
  2. top命令反应的是系统进程动态信息,默认10s更新一次。
  3. ps和top都是从/proc目录下读取进程的状态信息,内核把当前系统进程的各种有用信息都放在这个伪目录下。

常见ps命令:
ps -aux: 查看系统所有进程
ps -l: 进查看自己的bash相关进程

top 命令详解,请参考http://www.jb51.net/article/40807.htm
基本命令:

  • 按键盘数字“1”,可监控每个逻辑CPU的状况:
  • 键盘“b”(打开/关闭加亮效果)
  • 键盘“x”(打开/关闭排序列的加亮效果)
  • ”shift + >”或”shift + <”可以向右或左改变排序列
  • 敲击“f”键,编排基本视图中的显示字段
  • s,改变画面更新频率
  • l,关闭或开启第一部分第一行 top 信息的表示
  • t,关闭或开启第一部分第二行 Tasks 和第三行 Cpus 信息的表示
  • m,关闭或开启第一部分第四行 Mem 和 第五行 Swap 信息的表示

具体使用方法,可以使用 man [cmd] 查看。

相关参数说明

VSZ & VIRT

  1. 进程使用的虚拟内存值总量,包括所有代码,数据,共享库已经被swapped out的。VIRT = SWAP + RES。
  2. VSZ来自ps命令, VIRT来自top命令,二者均表示进程占用的虚拟内存大小。
  3. 假如进程申请100m的内存,但实际只使用了10m,那么它会增长100m,而不是实际的使用量

RES & RSS

  1. 进程当前使用的内存大小,但不包括swap out. RES = CODE +DATA。
  2. 包含其他进程的共享
  3. RES 来自 top 命令, RSS 来自 ps 命令,两者在表示意义上没有区别,都是从 /cat/proc/[pid]/stat 文件中读取的信息。
  4. 如果申请100m的内存,实际使用10m,它只增长10m,与VIRT相反
  5. 关于库占用内存的情况,它只统计加载的库文件所占内存大小

CODE
可执行代码占用的物理内存大小

DATA

  1. 物理内存中存放数据的大小,在程序运行中需要用到
  2. 如果top命令没有显示, 按f键显示

SHR

  1. 共享内存大小
  2. 除了自身进程的共享内存,也包含其他进程的共享内存
  3. 计算某个进程所占用物理内存的大小: RES - SHR
  4. swap out后,该值会下降。

查看一个进程的内存信息步骤

  1. 获取进程PID
 
    
1
2
 
    
$ ps -aux | grep /usr/sbin/NetworkManager
root 845 0.0 0.0 387084 13332 ? Ssl 3月28 0:00 /usr/sbin/NetworkManager --no-daemon
  1. 查看进程的所有线程

     
          
    1
    2
    3
    4
    5
    6
     
          
    $ ps mp 845 -o THREAD,tid
    USER %CPU PRI SCNT WCHAN USER SYSTEM TID
    root 0.0 - - - - - -
    root 0.0 19 - - - - 845
    root 0.0 19 - - - - 1025
    root 0.0 19 - - - - 1027
  2. 查看所有子进程

     
          
    1
    2
    3
    4
    5
     
          
    $ pstree -p 845
    NetworkManager(845)─┬─dhclient(30278)
    ├─dnsmasq(1123)
    ├─{gdbus}(1027)
    └─{gmain}(1025)

使用 /proc 下文件

/proc/[pid]/ 下面与进程内存相关的文件主要有maps , smaps, status
maps: 文件可以查看某个进程的代码段、栈区、堆区、动态库、内核区对应的虚拟地址
smaps: 显示每个分区更详细的内存占用数据
status: 包含了所有CPU活跃的信息,该文件中的所有值都是从系统启动开始累计到当前时刻

有名与匿名:
一个文件可以映射到进程的一段内存区域中,映射的文件描述符保存在vm_area_struct->vm_file域中,这种内存区域叫做有名内存区域,相反,属于匿名映射内存区域。

maps 文件分析

Proc/[pid]/maps 显示进程映射了的内存区域和访问权限。对应内核中的操作集为 proc_pid_maps_op,具体的导出函数为 show_map 。内核中进程的一段地址空间用一个vm_area_struct结构体表示,所有地址空间存储在task->mm->mmap链表中。

截取一行内容如下:

 
    
1
 
    
7f4e3f5ca000-7f4e3f674000 r-xp 00000000 08:02 525202 /usr/lib/x86_64-linux-gnu/NetworkManager/libnm-device-plugin-wifi.so

Vm_area_struct每项对应解析如下表所示:

vm_area_struct项maps项含义
vm_start“-”前一列,如7f4e3f5ca000此段虚拟地址空间起始地址
vm_end“-”后一列,如7f4e3f674000此段虚拟地址空间结束地址
vm_flags第三列,如r-xp此段虚拟地址空间的属性。r:读,w:写,x:执行,p和s共用一个字段,互斥关系,p表示私有段,s表示共享段,如果没有相应权限,则用’-’代替
vm_pgoff第四列,如00000000对有名映射,表示此段虚拟内存起始地址在文件中以页为单位的偏移。对匿名映射,它等于0或者vm_start/PAGE_SIZE
vm_file->f_dentry->d_inode->i_sb->s_dev第五列,如08:02映射文件所属设备号。对匿名映射l来说,因为没有文件在磁盘上,所以没有设备号,始终为00:00。对有名映射来说,是映射的文件所在设备的设备号
vm_file->f_dentry->d_inode->i_ino第六列,如525202映射文件所属节点号。对匿名映射来说,因为没有文件在磁盘上,所以没有节点号,始终为00:00。对有名映射来说,是映射的文件的节点号
 第七列,如/lib/ld-2.5.so对有名来说,是映射的文件名。对匿名映射来说,是此段虚拟内存在进程中的角色。[stack]表示在进程中作为栈使用,[heap]表示堆。其余情况则无显示

maps文件只能显示简单的分区,smap文件可以显示每个分区的更详细的内存占用数据。

smap 文件分析

截取一段文件,各字段解析如下:

 
    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
 
    
7f148b2fa000-7f148b2fb000 rw-p 00026000 08:02 2883675 /lib/x86_64-linux-gnu/ld-2.23.so
Size: 4 kB 虚拟内存大小
Rss: 4 kB 实际使用物理内存大小 RSS = Shared_Clean+Shared_Dirty+Private_Clean+Private_Dirty
Pss: 4 kB RSS中私有的内存页面
Shared_Clean: 0 kB RSS中共享内存,没有被改写的页面
Shared_Dirty: 0 kB RSS中共享内存,被改写的页面
Private_Clean: 0 kB RSS中私有内存,未被改写
Private_Dirty: 4 kB RSS中私有内存,被改写
Referenced: 4 kB
Anonymous: 4 kB
AnonHugePages: 0 kB
Shared_Hugetlb: 0 kB
Private_Hugetlb: 0 kB
Swap: 0 kB 处于交换区的页面大小
SwapPss: 0 kB
KernelPageSize: 4 kB 操作系统一个页面大小
MMUPageSize: 4 kB 体系结构MMU一个页面大小
Locked: 0 kB
VmFlags: rd wr mr mw me dw ac sd

Dirty页面如果没有交换机制的情况下,应该是不能回收的。

分析脚本:
自己写了个简单的分析脚本,如下,可以根据需要进行修改。

 
    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
 
    
#! /bin/bash
awk 'BEGIN{
total = 0;
printf("SIZE\tRSS\tSHARED_CLEAN\tSHARED_DIRTY\tPRIVATE_CLEAN\tPRIVATE_DIRTY\n")
}{
if(NF >3){
if($2 ~ /[r-][w-][x-][ps]/){
if($6 =="")
name = $1;
else
name = $6;
}
}
while(getline)
{
if(NF >3){
if($2 ~ /[r-][w-][x-][ps]/){
if($6 =="")
name = $1;
else
name = $6;
}
}
if($1 ~ /^Size/){
size = $2;
total += $2;
}
if($1 ~ /Rss/){
rss = $2;
}
if($1 ~ /Shared_Clean/){
shared_clean = $2;
}
if($1 ~ /Shared_Dirty/){
shared_dirty = $2;
}
if($1 ~ /Private_Clean/){
private_clean = $2;
}
if($1 ~ /Private_Dirty/){
private_dirty = $2;
}
if($1 ~ /VmFlags/){
printf("%d\t%d\t%d\t%d\t%d\t%d\t%s\n",size,rss,shared_clean,shared_dirty,private_clean,private_dirty,name);
size = 0;
name = "";
rss = 0;
shared_clean = 0;
shared_dirty = 0;
private_clean = 0;
private_dirty = 0;
continue;
}
}
}END{
printf("====total: %d\n", total);
}' $1

关于匿名映射
smaps 中可能会存在大量的匿名区域,它们是使用 mmap 机制生成的,但是没有关联到任何一个文件。通常情况下,它们主要用于处理一些琐碎的任务,比如处理没有在堆中申请的共享内存或者缓冲区。比如 pthread 使用匿名映射区作为新线程的栈,在linux 下,pthread 为 新线程申请 8M 空间作为栈空间 已经一个小小片内存(如4Kb)用于检测内存溢出。所以在每个pthread创建时,会分配一个映射到节点0的8Mb内存,和一个映射到节点0的4Kb区域。

status 文件分析

截取文件,解析个字段如下:

 
    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
 
    
Develop>cat /proc/24475/status
Name: netio 可执行程序的名字
State: R (running) 任务状态,运行/睡眠/僵死
Tgid: 24475 线程组号
Pid: 24475 进程id
PPid: 19635 父进程id
TracerPid: 0
Uid: 0 0 0 0
Gid: 0 0 0 0
FDSize: 256 该进程最大文件描述符个数
Groups: 0
VmPeak: 6330708 kB 内存使用峰值
VmSize: 268876 kB 进程虚拟地址空间大小
VmLck: 0 kB 进程锁住的物理内存大小,锁住的物理内存无法交换到硬盘
VmHWM: 16656 kB
VmRSS: 11420 kB 进程正在使用的物理内存大小
VmData: 230844 kB 进程数据段大小
VmStk: 136 kB 进程用户态栈大小
VmExe: 760 kB 进程代码段大小
VmLib: 7772 kB 进程使用的库映射到虚拟内存空间的大小
VmPTE: 120 kB 进程页表大小
VmSwap: 0 kB
Threads: 5 共享使用该信号描述符的任务的个数,在POSIX多线程序应用程序中,线程组中的所有线程使用同一个信号描述符。
SigQ: 0/63346 待处理信号的个数
SigPnd: 0000000000000000 屏蔽位,存储了该线程的待处理信号
ShdPnd: 0000000000000000 屏蔽位,存储了该线程组的待处理信号
SigBlk: 0000000000000000 存放被阻塞的信号
SigIgn: 0000000001000000 存放被忽略的信号
SigCgt: 0000000180000000 存放被俘获到的信号
CapInh: 0000000000000000 能被当前进程执行的程序的继承的能力
CapPrm: ffffffffffffffff 进程能够使用的能力,可以包含CapEff中没有的能力,这些能力是被进程自己临时放弃的,CapEff是CapPrm的一个子集,进程放弃没有必要的能力有利于提高安全性
CapEff: ffffffffffffffff 进程的有效能力
CapBnd: ffffffffffffffff
Cpus_allowed: 01
Cpus_allowed_list: 0
Mems_allowed: 01
Mems_allowed_list: 0
voluntary_ctxt_switches: 201
nonvoluntary_ctxt_switches: 909

meminfo 文件分析

整机内存使用情况的文件/proc/meminfo,摘取分析如下

 
    
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
 
    
Develop>cat /proc/meminfo
MemTotal: 8112280 kB 所有可用RAM大小 (即物理内存减去一些预留位和内核的二进制代码大小)
MemFree: 4188636 kB LowFree与HighFree的总和,被系统留着未使用的内存
Buffers: 34728 kB 用来给文件做缓冲大小
Cached: 289740 kB 被高速缓冲存储器(cache memory)用的内存的大小
(等于 diskcache minus SwapCache )
SwapCached: 0 kB 被高速缓冲存储器(cache memory)用的交换空间的大小
已经被交换出来的内存,但仍然被存放在swapfile中。
用来在需要的时候很快的被替换而不需要再次打开I/O端口
Active: 435240 kB 在活跃使用中的缓冲或高速缓冲存储器页面文件的大小,
除非非常必要否则不会被移作他用
Inactive: 231512 kB 在不经常使用中的缓冲或高速缓冲存储器页面文件的大小,可能被用于其他途径.
Active(anon): 361252 kB
Inactive(anon): 120688 kB
Active(file): 73988 kB
Inactive(file): 110824 kB
Unevictable: 0 kB
Mlocked: 0 kB
SwapTotal: 0 kB 交换空间的总大小
SwapFree: 0 kB 未被使用交换空间的大小
Dirty: 0 kB 等待被写回到磁盘的内存大小
Writeback: 0 kB 正在被写回到磁盘的内存大小
AnonPages: 348408 kB 未映射页的内存大小
Mapped: 33600 kB 已经被设备和文件等映射的大小
Shmem: 133536 kB
Slab: 55984 kB 内核数据结构缓存的大小,可以减少申请和释放内存带来的消耗
SReclaimable: 25028 kB 可收回Slab的大小
SUnreclaim: 30956 kB 不可收回Slab的大小(SUnreclaim+SReclaimable=Slab)
KernelStack: 1896 kB 内核栈区大小
PageTables: 8156 kB 管理内存分页页面的索引表的大小
NFS_Unstable: 0 kB 不稳定页表的大小
Bounce: 0 kB
WritebackTmp: 0 kB
CommitLimit: 2483276 kB
Committed_AS: 1804104 kB
VmallocTotal: 34359738367 kB 可以vmalloc虚拟内存大小
VmallocUsed: 565680 kB 已经被使用的虚拟内存大小
VmallocChunk: 34359162876 kB
HardwareCorrupted: 0 kB
HugePages_Total: 1536 大页面数目
HugePages_Free: 0 空闲大页面数目
HugePages_Rsvd: 0
HugePages_Surp: 0
Hugepagesize: 2048 kB 大页面一页大小
DirectMap4k: 10240 kB
DirectMap2M: 8302592 kB

总结

linux下内存占用是一个比较复杂的概念,不能简单通过一个单一指标就判断某个程序“内存消耗”大小,因为进程所申请的内存不一定真正会被用到(malloc或mmap的实现)而且真正用到了的内存也不一定是只有该进程自己在用 (比如动态共享库)。我们应该根据具体需求选择合适的方式去分析内存占用。

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux内存占用可能是因为程序运行时占用内存过多,或者是由于系统资源的不足。建议使用linux命令top或者free查看系统内存的使用情况,找出占用内存最多的程序,并采取相应的措施优化。 对于Linux内存占用的问题,建议采取以下步骤进行排查和优化: 1.使用命令top或者free查看系统内存的使用情况。在终端中输入"top"或者"free -m"命令,可以查看当前系统中各个进程占用内存的情况,以及系统总内存、已用内存和可用内存等信息。 2.找出占用内存最多的进程。在top命令中,按下shift+m可以将进程按照内存占用量从到低排序。在free命令中,可以查看各个内存区域的使用情况,找出占用内存最多的区域。 3.分析进程或者内存区域的使用情况。根据进程或者内存区域的使用情况,分析其是否存在内存泄漏或者占用过多内存的情况。如果存在问题,需要对相应的程序进行优化。 4.释放不必要的内存。如果存在不必要的内存占用,可以通过释放内存来缓解内存占用的问题。在终端中输入"sudo sync && sudo echo 3 > /proc/sys/vm/drop_caches"命令,可以释放系统中的缓存。 5.调整系统内存管理策略。在一些情况下,可能需要调整系统内存管理策略来优化内存的使用效率。例如,可以修改内核参数vm.swappiness的值,来控制系统使用交换分区的频率。 总之,针对Linux内存占用的问题,需要通过分析进程或者内存区域的使用情况,以及调整系统内存管理策略等措施,来优化内存的使用效率。 Linux内存占用的原因可能有很多,例如程序的内存泄漏、网络应用的频繁使用、多个应用程序同时运行等。最好的办法是检查系统进程并终止不必要的进程,以减少内存占用Linux内存使用率过可能是因为以下原因: 1. 运行的程序占用了大量内存。可以使用 top 或 htop 等工具查看进程占用内存情况,并尝试关闭不必要的程序或调整它们的配置,以降低它们的内存占用。 2. 内存泄漏。如果程序存在内存泄漏,它们会持续占用系统内存,直到内存耗尽。可以使用 valgrind 等工具来检查程序是否存在内存泄漏,并及时修复。 3. 缓存和缓冲区。Linux 会将一部分内存用于文件缓存和网络缓冲区,以提系统性能。这些缓存并不会直接占用系统内存,但会在需要时释放。如果系统内存不足,Linux 会自动清理这些缓存,释放内存空间。 4. 内存碎片。当大量进程频繁申请和释放内存时,会产生内存碎片,导致系统无法分配大块的连续内存空间。可以使用内存碎片整理工具(如 malloc_trim)来解决这个问题。 5. 系统参数设置不当。如果系统参数设置不当,可能会导致内存过度使用。可以使用 sysctl 等工具来检查和调整系统参数。 ### 回答2: Linux内存占用的原因可能很多,以下是一些常见的原因和解决方法。 1. 内存泄露:程序中存在内存泄露,导致内存占用不断增加。可以使用工具如Valgrind等来检测和调试程序。 2. 进程爆炸:如果系统中有大量的进程存在,每个进程都占用一些内存,就会导致整体内存占用。可以使用命令如ps、top等来查看进程情况,并对不必要的进程进行杀掉或优化。 3. 缓存和缓冲区:Linux系统会通过缓存和缓冲区等机制来提升系统的性能,但这些机制也会占用一部分内存。可以通过一些命令如free、vmstat等来查看缓存和缓冲区的情况,并通过调整内核参数来优化。 4. 虚拟内存:如果物理内存不够用,系统会使用虚拟内存进行内存管理,但虚拟内存的实现会带来一定的性能损失。可以通过增加物理内存或调整虚拟内存的相关参数来优化。 5. 软件/服务:很多软件或服务本身就是内存占用的,如数据库、Web服务器等。可以通过调整软件或服务的配置来优化内存占用,例如调节缓存大小、减少并发连接数等。 总之,针对Linux内存占用的问题,需要综合考虑各种因素,找到合适的解决方法来优化。常用的工具如top、ps、free、vmstat、sar等都可以帮助我们了解系统的内存使用情况。同时,也需要关注系统的物理内存、交换分区、进程数、缓存和虚拟内存等参数,并加以调整和优化。 ### 回答3: Linux内存占用可能有多种原因,以下是一些可能的解决方案: 1. 确认是否为缓存导致的内存占用。在Linux中,USED的内存不一定是真正占用内存,而有可能是被分配给缓存。可以使用命令“free -m”查看内存占用情况,并注意查看buffers/cache的占用情况。如果buffers/cache的占用比例较,可以尝试使用“sync && echo 3 > /proc/sys/vm/drop_caches”释放缓存。 2. 关闭不必要的服务。可以使用命令“systemctl list-unit-files”查看所有已安装的服务,并关闭不必要的服务来释放内存。 3. 开启内存垃圾回收机制。如果内存占用,可以尝试开启内存垃圾回收机制,将不常用的内存删除以释放内存。 4. 限制进程内存使用。可以使用“ulimit -v”命令限制进程使用的内存大小,避免单个进程占用过多内存。 5. 调整Swap空间。Swap空间可以作为虚拟内存使用,在内存不足时可以将部分数据写入Swap空间。可以调整Swap空间的大小来优化内存使用。 总结:如果Linux内存占用,首先需要确认是真正的内存占用还是缓存占用。如果是缓存占用,可以释放缓存;如果是真正的内存占用,可以调整服务、开启内存垃圾回收机制、限制进程内存使用等方式来优化内存使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值