Linux中nohup和&的用法和区别
在Linux执行任务时,如果键入Ctrl+C退出进行其他任务或者关闭当前session
当前任务就会终止 要想不让进程停止或者让进程在后台运行,就需要一些命令,nohup和&就是一种非常好的方式
首先以执行一个python脚本为例:
python test.py
nohup
和&
的区别
&:后台运行,但当用户退出(挂起)的时候,命令自动也跟着退出
什么意思呢? 意思是说, 当你在执行 python test.py &
的时候, 即使你用ctrl C
, 那么python test.py
照样运行(因为对SIGINT信号免疫)。 但是要注意, 如果你直接关掉shell后, 那么, 这个python进程同样消失。 可见, &的后台并不硬(因为对SIGHUP信号不免疫)。
nohup: 即no hang up,不挂断的运行
nohup的意思是忽略SIGHUP信号, 所以当运行nohup python test.py
的时候, 关闭shell, 那么这个python进程还是存在的(对SIGHUP信号免疫)。 但是, 要注意, 如果你直接在shell中用Ctrl C, 那么, 这个python进程也是会消失的(因为对SIGINT信号不免疫)
注意并没有后台运行的功能,就是指,用nohup运行命令可以使命令永久的执行下去,和用户终端没有关系,例如我们断开SSH连接都不会影响他的运行,注意了nohup没有后台运行的意思;&才是后台运行
综合使用
如果想让进程在后台不挂断的运行,需要nohup
和&
结合起来使用
nohup nohup python test.py &> /var/log/python.log &
nohup语法:
nohup Command [ Arg ... ] [ & ]
nohup 命令运行由 Command参数和任何相关的 Arg参数指定的命令,忽略所有挂断(SIGHUP)信号。在注销后使用 nohup 命令运行后台中的程序。要运行后台中的 nohup 命令,添加 & ( 表示“and”的符号)到命令的尾部。
如果不将 nohup 命令的输出重定向,输出将附加到当前目录的nohup.out
文件中。如果当前目录的 nohup.out 文件不可写,输出重定向到 $HOME/nohup.out 文件中。如果没有文件能创建或打开以用于追加,那么 Command 参数指定的命令不可调用。如果标准错误是一个终端,那么把指定的命令写给标准错误的所有输出作为标准输出重定向到相同的文件描述符。
实战讲解
首先准备一个python测试代码,是一个输出HelloWorld和数字的死循环脚本,每输出一行就等待1秒:
#!/usr/bin/python
# -*- coding: UTF-8 -*-
import time
var = 1
while var > 0 : # 该条件永远为true,循环将无限执行下去
print "HelloWorld: ", var
var = var +1
time.sleep(1) #每输出一行就休眠1秒
1. 前台运行
执行下列命令前台运行python脚本是什么情况呢?
python test.py
程序每隔一秒会在终端输出一个字符串。
此时如果键入Ctrl+C ,程序会收到一个SIGINT信号,如果不做特殊处理,程序的默认行为是终止(如上图)。
2. 后台运行
执行下面的命令在后台运行这个python脚本:
python test.py &
如上图,首先会在终端显示进程号是8778
如果键入Ctrl + C,发出SIGINT信号,程序会继续运行吗?
下图所示,键入Ctrl + C程序仍然会继续运行:
执行ps -ef|grep test.py
查询一下,程序进程确实存在,如下图所示:
此时如果关闭session即关闭xshell,程序会收到一个SIGHUP信号,此时会怎么样呢?
重新打开xshell的session,:
如上图,程序不会继续输出,而且执行ps -ef|grep test.py
刚刚打开的进程以及被关闭了。
3. 使用nohup运行脚本
如果使用nohup命令来运行python脚本的话,会是怎样的情况呢?
执行以下命令:
nohup python test.py
如上图,使用nohup 运行程序test.py,会发现:
- 前台没有出现进程号
- 有一个“忽略输入,输出至nohup.out”的提示
- hello的输出也没有出现在前台
如上图,在另一个xshell的session的窗口执行ps -ef|grep test.py
,会发现脚本已经在运行了,进程号是20085
在前一个图中提示输出至nohup.out的提示,那么我们在新打开的窗口去看看这里面是啥。
vi nohup.out
如上图,脚本的日志会在这个文件输出。
此时,如果我们关闭原来执行脚本的xshell的session,程序会收到一个SIGHUP信号,程序会不会关闭呢?
如上图,我们在新打开的xshell创建执行ps -ef | grep test.py
发现,这个PID为20085的python进程还存在
此时只能通过kill命令来杀死进程了:
kill -9 20085
然后,再次ps一下,如下图,进程已经被杀掉了
此时重新使用nohup执行一下这个python脚本:nohup python test.py
然后键入Ctrl+C,并且执行ps -ef | grep test.py
查看一下进程:
如上图所示,键入Ctrl+C后,程序收到SIGINT信号后,进程直接关闭了
4. 后台不挂断运行
nohup
和&
一起使用,我们来看看是什么情况:
使用以下指令运行程序:
nohup python test.py &
如上图,使用nohup python test.py &
运行程序后,可以看到:
- 会在终端显示进程号是21503
- 也会有一个“忽略输入,输出至nohup.out”的提示
- 键入Ctrl + C,发送SIGINT信号,似乎没反应。
关闭session,发送SIGHUP信号,再打开一个session窗口ps一下:
如上图,ID为21503的进程依然存在,后续也只能用kill来关闭它。
结论
使用&后台运行程序:
- 结果会输出到终端
- 使用Ctrl + C发送SIGINT信号,程序免疫
- 关闭session发送SIGHUP信号,程序关闭
使用nohup运行程序:
- 结果默认会输出到nohup.out
- 使用Ctrl + C发送SIGINT信号,程序关闭
- 关闭session发送SIGHUP信号,程序免疫
使用nohup和&配合来启动程序:
- 同时免疫SIGINT和SIGHUP信号
最佳实践方案:
不要将信息输出到终端标准输出,标准错误输出,而要用日志组件将信息记录到日志里
nohup命令可以将日志输入到文件中
- 如果不指定输出文件,默认输出到当前目录下的
nohup.out
文件 - 如果当前目录的 nohup.out 文件不可写,输出重定向到 $HOME/nohup.out 文件中。
举个例子:
nohup python test.py > test.log 2>&1 &
nohup ping www.baidu.com > ping.log 2>&1 &
Linux输出重定向: