【Linux进阶命令 04】lsof (看看是谁动了我的文件?)

简介: 【Linux进阶命令 04】lsof (看看是谁动了我的文件?)

文章目录

一、简介

二、lsof语法

2.1 基本格式

2.2 选项

2.3 输出字段解释

三、常用 lsof 操作

3.1 查看某文件的相关进程

3.2 网络相关:-i

3.3 指定进程号打开的文件:-p

3.4 指定用户打开的文件:-u

3.5 某进程打开的文件:-c

3.6 复合查询

四、进阶用法

4.1用文本处理命令(grep\awk\sed)结合使用

4.2 更多实用选项

一、简介

lsof(list open files)是一个列出当前系统打开文件的工具。


它主要用来获取被进程打开文件的信息。


在 Linux 中,一切皆文件,lsof 命令可以查看所有已经打开了的文件,比如:普通文件,目录,特殊的块文件,管道,socket 套接字,设备,Unix 域套接字等等。同时,它还可以结合 grep 以及 ps 命令进行更多的高级搜索

打开的文件:

在Linux中,一切都是文件,当进程打开现有文件或创建新文件时,内核向进程返回一个文件描述符,文件描述符就是内核为了高效管理已被打开的文件所创建的索引,用来指向被打开的文件,所有执行 I/O 操作的系统调用都会通过文件描述符。因此,打开的文件指的是进程打开并使用的文件。


二、lsof语法

2.1 基本格式

lsof [OPTIONS] [--] [NAMES]

在没有任何选项的情况下,lsof 列出所有属于活动进程的打开文件(一般会很多、很多、很多)。

2.2 选项

lsof 命令有很多选项,下面是一些常用的选项(完整的字节看手册):


-a:表示 AND,用于指定多个条件。

-c <command>:列出指定命令所打开的文件。

-d <fd>:列出指定文件描述符所打开的文件。

-g <gid>:列出指定组 ID 所属的进程所打开的文件。

-i:列出所有打开了网络套接字(TCP 和 UDP)的进程。

-n:禁止解析 IP 地址和端口号。

-p <pid>:列出指定进程 ID 所打开的文件。

-u <username>:列出指定用户所属的进程所打开的文件。


2.3 输出字段解释

使用lsof命令后输出一般有9个字段(9列):

COMMAND:进程的名称
PID:进程标识符
USER:进程所有者
FD:文件描述符。主要有:
  cwd:应用程序当前工作目录,这是该应用程序启动的目录,除非它本身对这个目录进行更改
  txt:该类型的文件是程序代码,如应用程序二进制文件本身或共享库,如上列表中显示的 /sbin/init 程序
  lnn:库引用(AIX)
  err:FD 信息错误
  jld:监狱目录(FreeBSD)
  ltx:共享库文本(代码和数据)
  mxx:十六进制内存映射类型号 xx
  m86:DOS合并映射文件
  mem:内存映射文件
  mmap:内存映射设备
  pd:父目录
  rtd:根目录
  tr:内核跟踪文件(OpenBSD)
  v86:VP/ix 映射文件
  0:标准输出
  1:标准输入
  2:标准错误
  文件描述符后一般还跟着文件状态模式:
  r:只读模式
  w:写入模式
  u:读写模式
  空格:文件的状态模式为 unknow,且没有锁定
  -:文件的状态模式为 unknow,且被锁定
  同时在文件状态模式后面,还跟着相关的锁:
  N:对于未知类型的 Solaris NFS 锁
  r:文件部分的读锁
  R:整个文件的读锁
  w:文件的部分写锁
  W:整个文件的写锁
  u:任何长度的读写锁
  U:用于未知类型的锁
  x:用于部分文件上的 SCO OpenServer Xenix 锁
  X:用于整个文件上的 SCO OpenServer Xenix 锁
  space:无锁
TYPE:文件类型。常见的文件类型有:
  REG:普通文件
  DIR:表示目录
  CHR:表示字符类型
  BLK:块设备类型
  UNIX:UNIX 域套接字
  FIFO:先进先出队列
  IPv4(6):IPv4(6) 套接字
DEVICE:磁盘名称
SIZE:文件的大小或文件偏移量(以字节为单位)
NODE:索引节点
NAME:打开文件的确切名称

例如:

root@CQUPTLEI:~# lsof -i udp
COMMAND       PID            USER   FD   TYPE   DEVICE SIZE/OFF NODE NAME
systemd         1            root   37u  IPv4     1001      0t0  UDP *:sunrpc 
systemd         1            root   39u  IPv6     1007      0t0  UDP *:sunrpc 
rpcbind       696            _rpc    5u  IPv4     1001      0t0  UDP *:sunrpc 
rpcbind       696            _rpc    7u  IPv6     1007      0t0  UDP *:sunrpc 
systemd-n     801 systemd-network   19u  IPv4    22416      0t0  UDP CQUPTLEI:bootpc 
systemd-r     803 systemd-resolve   12u  IPv4    21343      0t0  UDP localhost:domain 
avahi-dae     820           avahi   12u  IPv4    24274      0t0  UDP *:mdns 
avahi-dae     820           avahi   13u  IPv6    24275      0t0  UDP *:mdns 

三、常用 lsof 操作

3.1 查看某文件的相关进程

root@CQUPTLEI:~# lsof /bin/bash
COMMAND       PID USER  FD   TYPE DEVICE SIZE/OFF   NODE NAME
mysqld_sa    1188 root txt    REG  252,2  1183448 132810 /usr/bin/bash
sh           3495 root txt    REG  252,2  1183448 132810 /usr/bin/bash
bash      1520268 root txt    REG  252,2  1183448 132810 /usr/bin/bash
bash      1520271 root txt    REG  252,2  1183448 132810 /usr/bin/bash
bash      1666912 root txt    REG  252,2  1183448 132810 /usr/bin/bash

3.2 网络相关:-i

-i 选项用于列出所有打开了网络套接字(TCP 和 UDP)的进程。


常用的用法:


-i:列出所有打开了网络套接字(TCP 和 UDP)的进程。

-i tcp:列出所有打开了 TCP 网络连接的进程。

-i :<port>:列出某个端口被哪些进程打开了。

-i @<ip_address>:列出某个 IP 地址被哪些进程打开了。

-u <username> -i:列出某个用户打开了哪些网络连接。

例:

root@CQUPTLEI:~# lsof -i :443
COMMAND PID USER   FD   TYPE DEVICE SIZE/OFF NODE NAME
nginx   978 root   10u  IPv4  26143      0t0  TCP *:https (LISTEN)
nginx   983  www   10u  IPv4  26143      0t0  TCP *:https (LISTEN)
nginx   984  www   10u  IPv4  26143      0t0  TCP *:https (LISTEN)

3.3 指定进程号打开的文件:-p

-p 选项用于列出指定进程 ID 所打开的文件。

root@CQUPTLEI:~# lsof -p 1
COMMAND PID USER   FD      TYPE             DEVICE SIZE/OFF       NODE NAME
systemd   1 root  cwd       DIR              252,2     4096          2 /
systemd   1 root  rtd       DIR              252,2     4096          2 /
systemd   1 root  txt       REG              252,2  1620224     136385 /usr/lib/systemd/systemd
systemd   1 root  mem       REG              252,2  1369384     134892 /usr/lib/x86_64-linux-gnu/libm-2.31.so
systemd   1 root  mem       REG              252,2   178528     136839 /usr/lib/x86_64-linux-gnu/libudev.so.1.6.17
systemd   1 root  mem       REG              252,2  1575112     138611 /usr/lib/x86_64-linux-gnu/libunistring.so.2.1.0

3.4 指定用户打开的文件:-u

-u 选项用于列出指定用户所属的进程所打开的文件。

root@CQUPTLEI:~# ls /home
lighthouse  ubuntu  www
root@CQUPTLEI:~# lsof -u www
lsof: WARNING: can't stat() fuse.gvfsd-fuse file system /run/user/134/gvfs
      Output information may be incomplete.
COMMAND     PID USER   FD      TYPE             DEVICE  SIZE/OFF   NODE NAME
nginx       983  www  cwd       DIR              252,2      4096      2 /
nginx       983  www  rtd       DIR              252,2      4096      2 /
nginx       983  www  txt       REG              252,2   5060432 912450 /www/server/nginx/sbin/nginx
nginx       983  www  mem       REG              252,2    239896 131210 /usr/lib/x86_64-linux-gnu/libnss_systemd.so.2
nginx       983  www  DEL       REG                0,1            26134 /dev/zero
nginx       983  www  DEL       REG                0,1            26133 /dev/zero
nginx       983  www  DEL       REG                0,1            26132 /dev/zero
nginx       983  www  DEL       REG                0,1            26131 /dev/zero

3.5 某进程打开的文件:-c

-c 选项用于列出指定命令所打开的文件,不需要进程的完整名称,可以使用进程名的一部分。

root@CQUPTLEI:~# lsof -c alist
COMMAND PID USER   FD      TYPE             DEVICE SIZE/OFF    NODE NAME
alist   941 root  cwd       DIR              252,2     4096 1041395 /opt/alist
alist   941 root  rtd       DIR              252,2     4096       2 /
alist   941 root  txt       REG              252,2 43084616 1041396 /opt/alist/alist
alist   941 root  mem-r     REG              252,2    32768 1041407 /opt/alist/data/data.db-shm
alist   941 root    0r      CHR                1,3      0t0       6 /dev/null
alist   941 root    1u     unix 0xffffa05a31b67000      0t0   26720 type=STREAM
alist   941 root    2u     unix 0xffffa05a31b67000      0t0   26720 type=STREAM

3.6 复合查询

(1)同时满足多个条件:-a

用户www打开的tcp 443端口,逻辑与(and)

root@CQUPTLEI:~# lsof -a -u   www -i tcp:443
COMMAND PID USER   FD   TYPE DEVICE SIZE/OFF NODE NAME
nginx   983  www   10u  IPv4  26143      0t0  TCP *:https (LISTEN)
nginx   984  www   10u  IPv4  26143      0t0  TCP *:https (LISTEN)

(2)满足每个条件

不使用-a时,多个选项之间的关系是或(or)

root@CQUPTLEI:~# lsof -c ssh -c alist

四、进阶用法

4.1用文本处理命令(grep\awk\sed)结合使用

例: 与grep命令结合使用

打印服务器所有tcp连接中,文件描述符为10的连接:

root@CQUPTLEI:~# lsof -i tcp | grep "10u"
alist         941            root   10u  IPv6    31063      0t0  TCP *:5244 (LISTEN)
nginx         978            root   10u  IPv4    26143      0t0  TCP *:https (LISTEN)
nginx         983             www   10u  IPv4    26143      0t0  TCP *:https (LISTEN)
nginx         984             www   10u  IPv4    26143      0t0  TCP *:https (LISTEN)
BT-Panel     2638            root   10u  IPv4 23149843      0t0  TCP CQUPTLEI:8888->internettl.org:52156 (ESTABLISHED)

|为管道符,表示将左边的输出作为右边的输入

例: 与awk命令结合使用

对Linux的dns服务器(named)打开的tcp连接,按照描述符升序打印

lsof -i tcp -n| awk '/named/{print $1,$2,$3,$4,$5,$6,$6,$8,$9}' | sort -n -k 4
root@CQUPTLEI:~# lsof -i tcp -n| awk '/named/{print $1,$2,$3,$4,$5,$6,$6,$8,$9}' | sort -n -k 4
named 945 bind 22u IPv4 29095 29095 TCP 127.0.0.1:953
named 945 bind 26u IPv4 26405 26405 TCP 127.0.0.1:domain
named 945 bind 27u IPv4 26405 26405 TCP 127.0.0.1:domain
named 945 bind 28u IPv4 26405 26405 TCP 127.0.0.1:domain
named 945 bind 32u IPv4 28479 28479 TCP 10.0.8.5:domain
named 945 bind 33u IPv4 28479 28479 TCP 10.0.8.5:domain
named 945 bind 34u IPv4 28479 28479 TCP 10.0.8.5:domain
named 945 bind 37u IPv6 26406 26406 TCP [::1]:domain
named 945 bind 38u IPv6 26406 26406 TCP [::1]:domain
named 945 bind 39u IPv6 26406 26406 TCP [::1]:domain
named 945 bind 42u IPv6 28513 28513 TCP [fe80::5054:ff:fe4b:ad85]:domain
named 945 bind 43u IPv6 28513 28513 TCP [fe80::5054:ff:fe4b:ad85]:domain
named 945 bind 44u IPv6 28513 28513 TCP [fe80::5054:ff:fe4b:ad85]:domain
named 945 bind 45u IPv6 29096 29096 TCP [::1]:953
named 945 bind 48u IPv4 36766 36766 TCP 172.17.0.1:domain
named 945 bind 49u IPv4 36766 36766 TCP 172.17.0.1:domain
named 945 bind 50u IPv4 36766 36766 TCP 172.17.0.1:domain

sort -n -k 4:对结果排序,按照第四列数字升序

4.2 更多实用选项

lsof还有很多选项,比如:对文件夹递归搜索

-?, -h 
  显示帮助信息
-a
参数被视为逻辑与 AND,会影响全部的参数
-A A
  在配置了 AFS 分布式网络文件系统的系统上可用,其 AFS 内核代码是通过动态模块实现的。通过 A 指定备用名称列表文件,在该文件中可以找到动态模块的内核地址
-b
  避免 lsof 因调用可能阻塞的内核函数而产生阻塞,比如 lstat(2)、readlink(2) 和 stat(2) 等内核函数
-c C
  显示出以字符或字符串 C 开头的命令程序开启的文件,如 lsof -c init。如果 C 以斜杠 / 开头和结尾,则斜杠之间的字符被解释为正则表达式。该选项可多次指定
+c W
  指定 COMMAND 列的宽度,单位字符。默认为 9
-C
  禁用从内核的名称缓存中报告任何路径名
-D D
  指导 lsof 使用设备缓存文件。该选项的使用有时受到限制。-D 必须后面跟着一个函数字母,函数字母后面可以有一个路径名称。lsof 识别以下功能字母:
  ? 报告设备缓存文件路径
  b 构建设备缓存文件
  i 忽略设备缓存文件
  r 读取设备缓存文件
  u 读取并更新设备缓存文件
+D D
  递归搜索目录 D。如显示在 /usr/local 及其子目录下被程序开启的文件:lsof +D /usr/local
+d D
   非递归搜索目录 D。如显示在 /usr/local 下被程序开启的文件:lsof +d /usr/local
-d FD
  指定文件描述符列表,可以采用逗号分隔,也可以指定范围。比如 1,2,3 或 1-3。如果前面包含尖号,表示排除。如显示 FD 为 4 的进程:lsof -d 4
+|-e S
  豁免(exempt)路径名称为 S 的文件系统不受可能阻塞的内核函数调用的影响。+e 选项豁免 stat(2)、lstat(2) 和大多数 readlink(2) 内核函数调用。-e 选项只能豁免 stat(2) 和 lstat(2) 内核函数调用
+|-E
  +E 指定使用端点信息显示 Linux 管道、Linux UNIX 套接字和 Linux 伪终端文件,并显示端点的文件。-E 则不显示端点的文件
-F LIST
  指定字符列表 LIST,选择输出给另一程序处理的字段,各字段对应的字符见下文
+|-f [cfgGn]
  f 本身澄清了路径名参数的解释方式。当后面跟着 c、f、g、G 或 n 时,它指定要启用(+)或抑制(-)内核文件结构信息。
  c 文件结构使用计数(not Linux)
  f 文件结构地址(not Linux)
  g 文件标志缩写(Linux 2.6.22 及更高版本)
  G 十六进制文件标志(Linux 2.6.22 及更高版本)
  n 文件结构节点地址(not Linux)
-g [PGID]
  选择或排除属于指定进程组的进程打开的文件。 进程组 ID 使用逗号分隔,如果 PGID 前面包含尖号,表示排除。若没有指定 PGID,则显示全部。如显示 PGID 为 6 和 7 的进程:lsof -g6,7
-i [I]
  选择其 Internet 地址与 -i 中指定的地址匹配的文件,若没有相关地址被指定,则监听全部。
  用法: lsof -i [46][protocol][@hostname|hostaddr][:serivce|port]
  说明:4 6 分别表示 IPv4 和 IPv6   
    protocol: TCP or UDP   
    hostname:主机名
    hostaddr:IPv4 或 IPv6 地址
    service:主机提供的服务的名称,即 /etc/services 中的 service name
    port:端口号
-K
  在支持任务(线程)报告方式的系统上输出进程的任务(线程)列表
-k K
  指定内核名称列表文件,代替 /vmunix、/mach 等
-l
  禁止将 user ID 转换为登录的名称,默认是登录名称
+|-L [L]
  + 或 - 表示开启或关闭显示文件连接数,如果只有单纯的 +L,后面没有任何数字,则表示显示全部,如果后面有数字,只有文件连接数少于该数字的会被列出
+|-m M
  -m 指定一个内核内存文件 M ,代替 /dev/kmem 或 /dev/mem。+m 将装载补充文件写入标准输出文件
+|-M
  启用或禁用报告本地 TCP、UDP 和 UDPLITE 端口的端口映射器注册
-n
  不将 IP 地址转换为主机名
-N
  显示 NFS 文件
-o
  始终显示文件偏移量。它导致 SIZE/OFF 输出列标题更改为 OFFSET
-o O
  指定在文件偏移量的 0t 之后要打印的小数位数
-O
  指示 lsof 避免被某些内核操作阻塞。即在分叉的子进程中执行它们。虽然使用此选项将减少 lsof 启动开销,但也可能导致 lsof 在内核不响应函数时挂起。谨慎使用此选项
-P
  禁止将网络文件的端口号转换为端口名
-p S
  排除或选择进程的文件列表,进程 ID 列表使用逗号分隔,如 123 或 123,^456。尖号表示排除指定 PID
-R
  使用列 PPID 列出父进程的 PID
+|-r [T[mFMT]]
  控制 lsof 不断重复执行,间隔 T 秒,默认为 15s。-r 永远不断地执行,直到收到中断讯号(ctrl+ c),+r 一直执行,直到没有文件被显示。可选的 mFMT 参数指定标记线的格式,格式 FMT 遵循 C 语言标准库函数 strftime(3) 的规范
-S [T]
  指定内核函数 lstat(2)、readlink(2) 和 stat(2) 的可选超时秒值,否则可能会死锁。t 的最小值是 2;默认值是 15
-s [P:S]
  列出文件的大小,若该文件没有大小,则留下空白。它导致 SIZE/OFF 输出列标题更改为 SIZE。P 表示协议名称 TCP  or  UDP,S 表示逗号分隔的协议状态
-T [T]
  -T 没有参数则禁用 TCP/TPI 信息报告。跟如下参数,则显示指定 TCP/TPI 信息:
  f 选择报告套接字选项,状态和值,以及 TCP标志和值
  q 选择队列长度
  s 选择连接状态
  w 选择窗口大小
-t
  生成只有进程标识符而没有标题的简洁输出,这样输出可以通过管道传递给 kill(1) 杀死
-U
  选择 UNIX 域套接字文件的列表
-u USERS
  选择登录名或用户 ID 位于逗号分隔集 USERS 中的用户的文件列表。如 root 或 548,root”,如果用户名或用户 ID 前有尖号 ^,表示排除
-V
  指示被要求列出但找不到的项
-v
    显示版本信息
+|-w
     启用(+)或禁用(-)警告消息
-X
  Linux 下跳过所有打开的 TCP、UDP 和 UDPLITE IPv4 和 IPv6 文件的信息报告
-x [fl]
  一般与选项 +d 和 +D 选项,指示搜索时是否跨文件系统和符号链接。-x 不跟任何参数时,表示跨文件系统和符号链接
-Z [Z]
  指定如何处理 SELinux 安全上下文。当在运行的 Linux 内核中禁用SELinux时,Z 字段将被抑制输出。-Z 选项不跟参数,如 -Z -,安全上下文将列在 SECURITY-CONTEXT 列中输出
--
  双减号表示选项结束
NAMES
  列出指定文件,符号链接在使用前将被解析
相关文章
|
1月前
|
Linux Shell 网络安全
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
本指南介绍如何利用 HTA 文件和 Metasploit 框架进行渗透测试。通过创建反向 shell、生成 HTA 文件、设置 HTTP 服务器和发送文件,最终实现对目标系统的控制。适用于教育目的,需合法授权。
73 9
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
|
23天前
|
Ubuntu Linux Go
golang编译成Linux可运行文件
本文介绍了如何在 Linux 上编译和运行 Golang 程序,涵盖了本地编译和交叉编译的步骤。通过这些步骤,您可以轻松地将 Golang 程序编译成适合 Linux 平台的可执行文件,并在目标服务器上运行。掌握这些技巧,可以提高开发和部署 Golang 应用的效率。
171 14
|
1月前
|
Linux Shell
Linux 10 个“who”命令示例
Linux 10 个“who”命令示例
80 14
Linux 10 个“who”命令示例
|
22天前
|
存储 NoSQL Linux
linux积累-core文件是干啥的
核心文件是Linux系统在程序崩溃时生成的重要调试文件,通过分析核心文件,开发者可以找到程序崩溃的原因并进行调试和修复。本文详细介绍了核心文件的生成、配置、查看和分析方法
70 6
|
1月前
|
Ubuntu Linux
Linux 各发行版安装 ping 命令指南
如何在不同 Linux 发行版(Ubuntu/Debian、CentOS/RHEL/Fedora、Arch Linux、openSUSE、Alpine Linux)上安装 `ping` 命令,详细列出各发行版的安装步骤和验证方法,帮助系统管理员和网络工程师快速排查网络问题。
142 20
|
24天前
|
Linux
linux查看目录下的文件夹命令,find查找某个目录,但是不包括这个目录本身?
通过本文的介绍,您应该对如何在 Linux 系统中查看目录下的文件夹以及使用 `find` 命令查找特定目录内容并排除该目录本身有了清晰的理解。掌握这些命令和技巧,可以大大提高日常文件管理和查找操作的效率。 在实际应用中,灵活使用这些命令和参数,可以帮助您快速定位和管理文件和目录,满足各种复杂的文件系统操作需求。
67 8
|
24天前
|
存储 NoSQL Linux
linux之core文件如何查看和调试
通过设置和生成 core 文件,可以在程序崩溃时获取详细的调试信息。结合 GDB 等调试工具,可以深入分析 core 文件,找到程序崩溃的具体原因,并进行相应的修复。掌握这些调试技巧,对于提高程序的稳定性和可靠性具有重要意义。
160 6
|
1月前
|
网络协议 Linux 应用服务中间件
kali的常用命令汇总Linux
kali的常用命令汇总linux
70 7
|
2月前
|
Linux 数据库
Linux中第一次使用locate命令报错?????
在Linux CentOS7系统中,使用`locate`命令时出现“command not found”错误,原因是缺少`mlocate`包。解决方法是通过`yum install mlocate -y`或`apt-get install mlocate`安装该包,并执行`updatedb`更新数据库以解决后续的“can not stat”错误。
49 9
|
2月前
|
监控 网络协议 Linux
Linux netstat 命令详解
Linux netstat 命令详解