来自于这篇文章:https://www.ibm.com/developerworks/cn/linux/l-cn-nohup/
- 首先我们在后台运行一个任务
[root@training ~]# ping www.baidu.com >/dev/null 2>&1 &
[root@training ~]# ps -ef | grep ping
root 7699 7243 0 20:04 pts/1 00:00:00 ping www.baidu.com
root 7701 7243 0 20:05 pts/1 00:00:00 grep ping
[root@training ~]# ps -ef | grep 7243
root 7243 7239 0 19:59 pts/1 00:00:00 -bash
root 7699 7243 0 20:04 pts/1 00:00:00 ping www.baidu.com
root 7873 7243 0 20:07 pts/1 00:00:00 ps -ef
root 7874 7243 0 20:07 pts/1 00:00:00 grep 7243
[root@training ~]# ps -ef | grep 7239
root 7239 14432 0 19:59 ? 00:00:00 sshd: root@pts/1
root 7243 7239 0 19:59 pts/1 00:00:00 -bash
root 7914 7243 0 20:08 pts/1 00:00:00 grep 7239
[root@training ~]# ps -ef | grep 14432
root 7239 14432 0 19:59 ? 00:00:00 sshd: root@pts/1
root 8004 7243 0 20:08 pts/1 00:00:00 grep 14432
root 14432 1 0 Dec06 ? 00:00:00 /usr/sbin/sshd
[root@training ~]# ps -ef | grep init
root 1 0 0 Nov08 ? 00:00:00 /sbin/init
ping这个后台进程的PID是7699,其父进程是7243
7243这个进程是-bash,其父进程是7239
7239这个进程是sshd: root@pts/1 ,其父进程是14432
14432这个进程是/usr/sbin/sshd,其父进程是1
1这个进程就是/sbin/init
通过pstree这个命令我们可以看得更清楚
init─┬─abrtd
├─acpid
├─agetty
├─sshd───sshd───bash─┬─ping
│ └─pstree
/sbin/init产生了/usr/sbin/sshd,
/usr/sbin/sshd产生了一个虚拟终端pts/1,
pts/1给出了bash这个shell,
然后我们在bash上运行ping这个进程
当 ssh 断开连接时,HUP 信号自然会影响到它下面的所有子进程(包括我们新建立的 ping 进程)。
- 那么当用户注销或者网络断开的时候,终端会收到HUP(hangup信号),从而关闭所有子进程,
因此,我们的解决办法就有两种途径:要么让进程忽略 HUP 信号,要么让进程运行在新的会话里从而成为不属于此终端的子进程。
- 让进程忽略 HUP 信号
[root@training ~]# nohup ping www.baidu.com >/dev/null 2>&1 &
[root@training ~]# ps -ef | grep ping
root 10605 7243 0 20:45 pts/1 00:00:00 ping www.baidu.com
root 10616 7243 0 20:45 pts/1 00:00:00 grep ping
nohup能通过忽略 HUP 信号来使我们的进程避免中途被中断,这样sshd被kill之后,ping变成孤儿进程,就会被init收养,此时的pstree为
init─┬─abrtd
├─acpid
├─agetty
├─ping
├─sshd───sshd───bash───pstree
- 让进程运行在新的会话里从而成为不属于此终端的子进程。
[root@training ~]# setsid ping www.baidu.com >/dev/null 2>&1 &
[root@training ~]# jobs
[1]+ Done setsid ping www.baidu.com > /dev/null 2>&1
[root@training ~]# ps -ef | grep ping
root 11061 1 0 20:51 ? 00:00:00 ping www.baidu.com
root 11103 7243 0 20:51 pts/1 00:00:00 grep ping
setsid之后马上运行jobs的话,可以看到这个后台进程,但是再运行jobs的话看不到任何进程了。
- &
这里还有一个关于 subshell 的小技巧。我们知道,将一个或多个命名包含在“()”中就能让这些命令在子 shell 中运行中,从而扩展出很多有趣的功能。
[root@training ~]# (ping www.baidu.com >/dev/null 2>&1 &)
[root@training ~]# ps -ef | grep ping
root 11652 1 0 20:59 pts/1 00:00:00 ping www.baidu.com
root 11697 7243 0 20:59 pts/1 00:00:00 grep ping
当我们将"&"也放入“()”内之后,我们就会发现所提交的作业并不在作业列表中,也就是说,是无法通过jobs来查看的。
从上例中可以看出,新提交的进程的父 ID(PPID)为1(init 进程的 PID),并不是当前终端的进程 ID。
因此并不属于当前终端的子进程,从而也就不会受到当前终端的 HUP 信号的影响了。
- disown
disown 能帮助我们来事后补救当前已经在运行了的作业,但是我并没有找到disown的yum源,在网上也没找到……
这里要注意的是:
CTRL+z可以将当前任务挂起,并放到后台,此时处于stopped状态,
bg %n可以让后台stopped的进程在后台继续进行,变为running,
fg %n可以将后台的线程拿到前台来执行
- screen
screen是一个很强大的工具,可以建立一个伪终端。
init─┬─abrtd
├─acpid
├─screen───bash───ping
├─sshd───sshd───bash───pstree
要注意的是:
在伪终端界面时,要通过CTRL+A+D来退出这个伪终端,回到原来的终端,参考这里:http://bg.artuion.com/linux/325.html