trap 是 linux Shell 中捕获并响应信号的核心机制,支持优雅退出、日志记录、资源清理、状态保存及信号转发;常用信号包括 int、TERM、EXIT、HUP;推荐用函数 封装 清理逻辑,注意子 shell 和 exec 对 trap 的影响。

Linux Shell 中的 trap 命令,是用来捕获并响应进程收到的信号的核心机制。它不只用于“优雅退出”,还能实现日志记录、资源清理、状态保存、甚至信号转发等实用功能。
trap 基本语法与常见信号
trap 的核心格式是:trap 'command' signal1 SIGNAL2 ……
其中 command 可以是任意合法的 shell 命令或函数调用,SIGNAL 可以是数字(如 2)或名称(如 INT、TERM、EXIT)。
常用信号包括:
- INT(2):Ctrl+C 触发,适合做中断清理
- TERM(15):默认 kill 发送的信号,用于常规终止
- EXIT:不是真实信号,而是 shell 退出前自动触发,最常用于兜底清理
- HUP(1):终端挂起或父进程退出时可能收到,适合守护进程重载配置
用函数封装清理逻辑,提升可读性与复用性
把复杂操作写进函数,再让 trap 调用,比直接拼接命令更清晰、易维护。
例如:
cleanup() { echo " 正在清理临时文件……" rm -f /tmp/myapp_$$ echo " 已释放锁 " rm -f /tmp/myapp.lock } trap cleanup INT TERM EXIT
注意:$$ 是当前 shell 进程 PID,可用于生成唯一临时名;EXIT 会确保无论正常结束还是被信号中断,清理都会执行。
慎用 signal 0 和子 shell 中的 trap 行为
trap '' 0 或 trap : 0 可屏蔽 EXIT,但容易引发意外——比如在子 shell((……))中设了 trap,其 作用域 仅限该子 shell,父 shell 不受影响。
常见陷阱:
- 在 for 循环 或管道中启动的子 shell,
trap EXIT不会传播过去 - 用
exec替换当前进程后,原有 trap 会被清除 - 捕获
CHLD时需配合wait -n避免僵尸进程,且不同 shell 对它的支持略有差异
实战技巧:信号转发 + 状态反馈
有时你希望先处理,再把信号传给子进程(比如管理后台服务)。可以这样写:
pid=0 start_service() { sleep 100 & pid=$!} trap 'echo " 收到 TERM,准备停止服务 "; kill "$pid" 2>/dev/null; wait "$pid" 2>/dev/null; exit 0' TERM start_service wait "$pid"
这段脚本能捕获 TERM,向子进程发信号、等待其退出,再自身退出,避免“假死”或孤儿进程。
不复杂但容易忽略。