要将一个命令放到后台执行,我们一般使用nohup sh command &
&都知道是放到后台执行这个命令,那么nohup是做什么的?
这就要从unix的信号说起,unix的信号机制可以说进程间通信的一种,进程间可以通过发送信号来完成某些特定的动作,比较熟悉就是kill -9 pid
先看linux有哪些信号:
[root@limt ~]# kill -l
1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL 5) SIGTRAP
6) SIGABRT 7) SIGBUS 8) SIGFPE 9) SIGKILL 10) SIGUSR1
11) SIGSEGV 12) SIGUSR2 13) SIGPIPE 14) SIGALRM 15) SIGTERM
16) SIGSTKFLT 17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP
21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO 30) SIGPWR
31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3
38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7
58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
63) SIGRTMAX-1 64) SIGRTMAX
[root@limt ~]# more /usr/include/bits/signum.h
#define SIGHUP 1 /* Hangup (POSIX). */
#define SIGINT 2 /* Interrupt (ANSI). */
#define SIGQUIT 3 /* Quit (POSIX). */
#define SIGILL 4 /* Illegal instruction (ANSI). */
#define SIGTRAP 5 /* Trace trap (POSIX). */
#define SIGABRT 6 /* Abort (ANSI). */
#define SIGIOT 6 /* IOT trap (4.2 BSD). */
#define SIGBUS 7 /* BUS error (4.2 BSD). */
#define SIGFPE 8 /* Floating-point exception (ANSI). */
#define SIGKILL 9 /* Kill, unblockable (POSIX). */
#define SIGUSR1 10 /* User-defined signal 1 (POSIX). */
#define SIGSEGV 11 /* Segmentation violation (ANSI). */
#define SIGUSR2 12 /* User-defined signal 2 (POSIX). */
#define SIGPIPE 13 /* Broken pipe (POSIX). */
#define SIGALRM 14 /* Alarm clock (POSIX). */
#define SIGTERM 15 /* Termination (ANSI). */
#define SIGSTKFLT 16 /* Stack fault. */
#define SIGCLD SIGCHLD /* Same as SIGCHLD (System V). */
#define SIGCHLD 17 /* Child status has changed (POSIX). */
#define SIGCONT 18 /* Continue (POSIX). */
#define SIGSTOP 19 /* Stop, unblockable (POSIX). */
#define SIGTSTP 20 /* Keyboard stop (POSIX). */
#define SIGTTIN 21 /* Background read from tty (POSIX). */
#define SIGTTOU 22 /* Background write to tty (POSIX). */
#define SIGURG 23 /* Urgent condition on socket (4.2 BSD). */
#define SIGXCPU 24 /* CPU limit exceeded (4.2 BSD). */
#define SIGXFSZ 25 /* File size limit exceeded (4.2 BSD). */
#define SIGVTALRM 26 /* Virtual alarm clock (4.2 BSD). */
#define SIGPROF 27 /* Profiling alarm clock (4.2 BSD). */
#define SIGWINCH 28 /* Window size change (4.3 BSD, Sun). */
#define SIGPOLL SIGIO /* Pollable event occurred (System V). */
#define SIGIO 29 /* I/O now possible (4.2 BSD). */
#define SIGPWR 30 /* Power failure restart (System V). */
#define SIGSYS 31 /* Bad system call. */
对于每种信号系统一般都有一个默认动作(一般是终止程序),然而除了SIGKILL,SIGSTOP 信号外,其他信号都可以被捕获并处理,
一个进程要向另外一个进程发生信号,可以通过kill -signal pid 或者调用函数kill发生
回到我们前面问题,为什么要nohup?因为我用使用Scrt这种终端工具退出的时候会向我们在当前shell下启动的进程发生一个SIGHUP信号,而SIGHUP信号的默认行为时终止进程,所以nohup的意思是屏蔽SIGHUP信号
下面我们做一个测试:
在一个窗口运行一个不带nohup的后台程序
[root@limt ~]# sh Testlsof.sh > 111.log &
[1] 4486
[root@limt ~]# jobs
[1]+ Running sh Testlsof.sh > 111.log &
在另外一个窗口查看后台程序
[root@limt ~]# ps -ef|grep Testlsof
root 4486 4315 0 20:11 pts/1 00:00:00 sh Testlsof.sh
root 4574 4500 0 20:12 pts/0 00:00:00 grep Testlsof
关闭第一个窗口,后台进程也退出
[root@limt ~]# ps -ef|grep Testlsof
root 4661 4500 0 20:12 pts/0 00:00:00 grep Testlsof 在一个窗口运行一个带nohup的后台程序
[root@limt ~]# nohup sh Testlsof.sh > 111.log &
[1] 2710
[root@limt ~]# nohup: 忽略输入重定向错误到标准输出端
[root@limt ~]# jobs
[1]+ Running nohup sh Testlsof.sh > 111.log &
在另外一个窗口查看后台程序
[root@limt ~]# ps -ef|grep Testlsof
root 2710 2664 0 20:23 pts/1 00:00:00 sh Testlsof.sh //父进程为shell
root 2794 2728 0 20:23 pts/2 00:00:00 grep Testlsof
关闭第一个窗口,后台进程没有退出
[root@limt ~]# ps -ef|grep Testlsof
root 2710 1 0 20:23 ? 00:00:00 sh Testlsof.sh <span style="font-family: Arial, Helvetica, sans-serif;">//父进程为init进程</span>
root 3223 2728 0 20:23 pts/2 00:00:00 grep Testlsof
怎么证明关闭终端发生的是SIGHUP信号?可以使用trap命令屏蔽SIGHUP,SIGHUP的值为1,使用如下命令:
[root@limt ~]# trap "" 1
然后在一个窗口运行一个不带nohup的后台程序
[root@limt ~]# sh Testlsof.sh > 111.log &
[1] 4068
[root@limt ~]# jobs
[1]+ Running sh Testlsof.sh > 111.log &
在另外一个窗口查看后台程序
[root@limt ~]# ps -ef|grep Test
root 4068 2728 0 20:26 pts/2 00:00:00 sh Testlsof.sh
root 4096 3963 0 20:26 pts/0 00:00:00 grep Test
关闭第一个窗口,后台进程没有退出
[root@limt ~]# ps -ef|grep Test
root 4068 1 0 20:26 ? 00:00:00 sh Testlsof.sh //当第一个窗口关闭后,它的父进程为1
root 4180 3963 0 20:26 pts/0 00:00:00 grep Test
如果你使用的是csh,那么就无需使用nohup命令,因为csh对SIGHUP做了处理
[root@limt ~]# csh
[root@limt ~]# sh Testlsof.sh > 111.log &
[1] 4602
[root@limt ~]# jobs
[1] + Running sh Testlsof.sh > 111.log 在另外一个窗口查看后台程序
[root@limt ~]# ps -ef|grep Testlsof
root 4602 4509 0 20:31 pts/1 00:00:00 sh Testlsof.sh
root 4658 4524 0 20:32 pts/0 00:00:00 grep Testlsof
关闭第一个窗口,后台进程没有退出
[root@limt ~]# ps -ef|grep Testlsof
root 4602 1 0 20:31 ? 00:00:00 sh Testlsof.sh
root 4676 4524 0 20:32 pts/0 00:00:00 grep Testlsof