nohup
SIGHUP
信号
SIGHUP
会在以下3种情况下被发送给相应的进程:
- 终端关闭时,该信号被发送到session首进程以及作为job提交的进程(即用 & 符号提交的进程);
- session首进程退出时,该信号被发送到该session中的前台进程组中的每一个进程;
- 若父进程退出导致进程组成为孤儿进程组,且该进程组中有进程处于停止状态(收到
SIGSTOP
或SIGTSTP
信号),该信号会被发送到该进程组中的每一个进程。
例如:在我们登录Linux时,系统会分配给登录用户一个终端(Session)。在这个终端运行的所有程序,包括前台进程组和后台进程组,一般都属于这个 Session。当用户退出Linux登录时,前台进程组和后台有对终端输出的进程将会收到SIGHUP
信号。这个信号的默认操作为终止进程,因此前台进 程组和后台有终端输出的进程就会中止。
nohup 命令
nohup
不会断开与终端的命令,它会使您的脚本被忽略SIGHUP
,并重定向stdout/stderr
到一个文件nohup.out
,以便在您注销后该命令可以在后台继续运行。
nohup
不会自动将命令在后台运行。必须通过用 &
来明确地做到后台运行。
jobs
可以打印当前正在运行的作业及其状态。如果命令jobs
找不到它,那么它不再是该shell的子进程。
fg
即使运行后,也可以使用bash将后台作业带回前台nohup
。但这不会改变输出重定向,它仍将转移到文件中nohup.out
。
测试
防止退出登录后,脚本结束运行,使用nohup
。
#!/bin/bash
for i in `seq 1000`;
do
echo "$i";
sleep 2;
done;
nohup sh t.sh 2>&1 &
nohup
与&
通常,在运行命令后使用&
和退出shell时,shell将使用hangup signal(kill -SIGHUP <pid>)
终止子命令。这可以使用nohup
,因为它捕获信号并忽略它,以便它永远不会到达实际的应用程序。
如果您正在使用bash,则可以使用该命令shopt | grep hupon
查明您的shell是否将SIGHUP
发送到其子进程。如果它关闭,进程将不会终止。
当huponexit
为off
时,exit时不会向终端所属任务发SIGHUP
信号,激活该选项:shopt -s huponexit
将已经运行的进程放到nohup下
Ctrl+ Z
停止(暂停)程序并返回shell。bg
在后台运行它。disown -h [job-spec]
其中[job-spec]是作业编号(类似于%1第一个正在运行的作业;使用jobs命令查找您的编号),以便在终端关闭时不会终止作业。
假设由于某种原因Ctrl+ Z也无法正常工作,请转到另一个终端,找到进程ID(使用ps)并运行
kill -SIGSTOP PID
kill -SIGCONT PID
SIGSTOP
将暂停该进程并用SIGCONT
在后台恢复该进程。所以现在,关闭两个终端并不会停止你的过程。