在 Linux 系统中,如果不小心误删除了应用程序正在使用的文件的话,可以在停止应用前通过 lsof 命令恢复被删除的文件。废话不多说,恢复过程见下文。
通过 ps 拿到进程号(假设应用程序为 appDemo):
- $ ps axuf |grep appDemo
- qileq 8932 4.9 13.2 64909644 43721472 ? Sl Jan20 867:59 /usr/java/default/bin/java com.qileq.appDemo
第二列的 8932 即进程号。
通过 lsof 查看指定进程被删除文件的文件描述符(假设被删除文件名为 app.log):
- $ lsof -p 8932 |grep app.log
- COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME
- java 8932 qileq 1w REG 253,0 166029303 169140713 /home/qileq/demo/app.log~ (deleted)
- java 8932 qileq 397w REG 253,0 166062569 169140713 /home/qileq/demo/app.log~ (deleted)
第四列即文件描述符,其值可能为如下两大类之一:
如果应用程序持有文件锁的话,则文件描述符还会使用如下字符表示持有的锁类型:
如常用的 FD 值有 8u、mem 等样式的值,稍微复杂的值有 mem-W、5uW 等类型。
/proc/PID/fd/FD 即为被删除的文件,如上述例子中的 /proc/8932/fd/1 即表示其中一个被删除的日志文件。通过 cp 命令恢复该文件:
- $ cp /proc/8932/fd/1 app.log.1
- $ stat app.log.1
- File: ‘app.log.1’
- Size: 166460709 Blocks: 325120 IO Block: 4096 regular file
- Device: fd00h/64768d Inode: 169140672 Links: 1
- Access: (0640/-rw-r-----) Uid: ( 1005/ qileq) Gid: ( 1005/ qileq)
- Access: 2023-02-01 18:11:22.337483269 +0800
- Modify: 2023-02-01 18:11:19.082499139 +0800
- Change: 2023-02-01 18:11:19.082499139 +0800
- Birth: -
待恢复所有被误删除的文件后,最好还是将应用重启下,以免新生成的文件可能也处于被删除状态。