在 Linux 系统中,检查文件是否被占用(被哪个进程打开)主要有两个常用的命令:lsof 和 fuser。
lsof 命令
lsof (List Open Files) 是最常用的工具,它可以列出当前系统打开的所有文件。
# lsof -h
lsof 4.87
latest revision: ftp://lsof.itap.purdue.edu/pub/tools/unix/lsof/
latest FAQ: ftp://lsof.itap.purdue.edu/pub/tools/unix/lsof/FAQ
latest man page: ftp://lsof.itap.purdue.edu/pub/tools/unix/lsof/lsof_man
usage: [-?abhKlnNoOPRtUvVX] [+|-c c] [+|-d s] [+D D] [+|-f[gG]] [+|-e s]
[-F [f]] [-g [s]] [-i [i]] [+|-L [l]] [+m [m]] [+|-M] [-o [o]] [-p s]
[+|-r [t]] [-s [p:s]] [-S [t]] [-T [t]] [-u s] [+|-w] [-x [fl]] [--] [names]
Defaults in parentheses; comma-separated set (s) items; dash-separated ranges.
-?|-h list help -a AND selections (OR) -b avoid kernel blocks
-c c cmd c ^c /c/[bix] +c w COMMAND width (9) +d s dir s files
-d s select by FD set +D D dir D tree *SLOW?* +|-e s exempt s *RISKY*
-i select IPv[46] files -K list tasKs (threads) -l list UID numbers
-n no host names -N select NFS files -o list file offset
-O no overhead *RISKY* -P no port names -R list paRent PID
-s list file size -t terse listing -T disable TCP/TPI info
-U select Unix socket -v list version info -V verbose search
+|-w Warnings (+) -X skip TCP&UDP* files -Z Z context [Z]
-- end option scan
+f|-f +filesystem or -file names +|-f[gG] flaGs
-F [f] select fields; -F? for help
+|-L [l] list (+) suppress (-) link counts < l (0 = all; default = 0)
+m [m] use|create mount supplement
+|-M portMap registration (-) -o o o 0t offset digits (8)
-p s exclude(^)|select PIDs -S [t] t second stat timeout (15)
-T qs TCP/TPI Q,St (s) info
-g [s] exclude(^)|select and print process group IDs
-i i select by IPv[46] address: [46][proto][@host|addr][:svc_list|port_list]
+|-r [t[m<fmt>]] repeat every t seconds (15); + until no files, - forever.
An optional suffix to t is m<fmt>; m must separate t from <fmt> and
<fmt> is an strftime(3) format for the marker line.
-s p:s exclude(^)|select protocol (p = TCP|UDP) states by name(s).
-u s exclude(^)|select login|UID set s
-x [fl] cross over +d|+D File systems or symbolic Links
names select named files or files on named file systems
Anyone can list all files; /dev warnings disabled; kernel ID check disabled.
基本语法
注意:建议加上 sudo,否则可能无法查看属于 root 或其他用户进程占用的文件信息。
示例:
假设你要检查 /var/log/nginx/access.log 是否被占用:
sudo lsof /var/log/nginx/access.log
输出解释:
如果文件被占用,你会看到类似下面的输出:
COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME
nginx 1234 root 4w REG 8,1 1024 123456 /var/log/nginx/access.log
nginx 1235 www 4w REG 8,1 1024 123456 /var/log/nginx/access.log
- COMMAND: 占用文件的进程名称(如 nginx)。
- PID: 进程 ID。
- USER: 运行该进程的用户。
- FD: 文件描述符(File Descriptor)。
- NAME: 文件名。
如果没有输出,说明该文件当前没有被任何进程打开。
常用语法
使用
lsof| more
lsof /bin/bash
lsof -u <username>
lsof -c nginx
lsof -p <pid>
lsof -i // 列出所有的网络连接
lsof -i tcp // 列出所有tcp 网络连接信息
lsof -i :80 // 列出谁在使用某个端口
lsof -a -u root -i // 列出某个用户的所有活跃的网络端口
# 查找指定目录下被打开的文件信息
lsof +d /var/log/nginx
# 递归查找指定目录中被打开的文件信息
lsof +D /var/log/nginx
fuser 命令
fuser 可以显示使用指定文件或文件系统的进程 ID。
基本语法
sudo fuser -v /path/to/file
常用参数:
-v: 详细模式(显示用户、命令等信息,不仅仅是 PID)。
-k: 直接杀掉占用该文件的进程(慎用)。
-i: 杀掉进程前询问确认。
示例
sudo fuser -v /var/log/syslog
输出示例:
USER PID ACCESS COMMAND
/var/log/syslog: syslog 856 F.... rsyslogd
可以看到 PID 为 856 的 rsyslogd 进程正在访问该文件。
实战
查看目录下被占用的所有文件
如果你想看某个目录下有哪些文件正在被占用:
使用 lsof:
sudo lsof +D /path/to/directory
使用 fuser:
sudo fuser -v /path/to/directory
查找已删除但仍被占用的文件
有时候你删除了一个大文件,但磁盘空间没有释放,通常是因为文件虽然在目录中删除了,但进程还在占用它。
查找命令:
这会列出所有标记为 (deleted) 但仍被打开的文件。你需要重启对应的服务或杀掉对应的 PID 才能真正释放空间。
总结
- 如果只是想看谁在用,用
lsof(信息更全)。
- 如果你想查看并杀死占用进程,用
fuser -k。
- 如果提示命令不存在,可能需要安装工具包(CentOS:
yum install lsof psmisc, Ubuntu: apt install lsof psmisc)。