nohup & 与 2>&1

原因

当我们在终端或控制台工作时,可能不希望由于运行一个作业而占住了屏幕,因为可能还有更重要的事情要做,比如阅读电子邮件。对于密集访问磁盘的进程,我们更希望它能够在每天的非负荷高峰时间段运行(例如凌晨)。为了使这些进程能够在后台运行,也就是说不在终端屏幕上运行,有几种选择方法可供使用。

&

**当在前台运行某个作业时,终端被该作业占据;可以在命令后面加上 & 实现后台运行。(以 job 的方式)**例如:

sh test.sh &

适合在后台运行的命令有 find、费时的排序及一些 shell 脚本。在后台运行作业时要当心:需要用户交互的命令不要放在后台执行,因为这样你的机器就会在那里傻等。不过,作业在后台运行一样会将结果输出到屏幕上,干扰你的工作。如果放在后台运行的作业会产生大量的输出,最好使用下面的方法把它的输出重定向到某个文件中:

command > out.file 2>&1 & 

这样,所有的标准输出和错误输出都将被重定向到一个叫做 out.file 的文件中。

底层原理

每一个命令行终端都是一个 shell 进程,你在这个终端里执行的程序实际上都是这个 shell 进程分出来的子进程。正常情况下,shell 进程会阻塞,等待子进程退出才重新接收你输入的新的命令。加上 & 号,只是让 shell 进程不再阻塞,立刻返回(return status 为 0),然后就可以继续响应你的新命令了。但是无论如何,你如果关掉了这个 shell 命令行端口,依附于它的所有子进程都会退出。

nuhup

使用 & 命令后,作业被提交到后台运行,当前控制台没有被占用,但是一但把当前控制台关掉(退出帐户时),作业就会停止运行。**nohup 命令可以在你退出帐户之后继续运行相应的进程。nohup 就是不挂起的意思(no hang up)。**该命令的一般形式为:

nohup command &

如果使用 nohup 命令提交作业,那么在缺省情况下该作业的所有输出都被重定向到一个名为 nohup.out 的文件中,除非另外指定了输出文件:

nohup command > myout.file 2>&1 &

使用了 nohup 之后,很多人就这样不管了,其实这样有可能在当前账户非正常退出或者结束的时候,命令还是自己结束了。所以在使用 nohup 命令后台运行命令之后,需要使用 exit 正常退出当前账户,这样才能保证命令一直在后台运行。

  • Ctrl+c:终止前台命令。
  • jobs:查看当前有多少在后台运行的命令。jobs -l 选项可显示所有任务的 PID,jobs 的状态可以是 running、stopped、Terminated。但是如果任务被终止了(kill),shell 从当前的 shell 环境已知的列表中删除任务的进程标识。

底层原理

nohup 是一个 POSIX 命令,用于忽略 SIGHUP ("signal hang up",挂断信号,是终端注销时所发送至程序的一个信号)。所以使用 nohup 后即使退出终端也不会是进程被关掉啦,因为它忽略了这个信号。

此外,还有一种方式是 (cmd &) 。这样是将 cmd 命令挂到一个 systemd 系统守护进程名下,认 systemd 做爸爸,这样当你退出当前终端时,对于刚才的 cmd 命令就完全没有影响了。

2>&1

2>&1 是将标准出错重定向到标准输出,这里的标准输出已经重定向到了 out.file 文件,即将标准出错也输出到 out.file 文件中。

试想 2>1 代表什么,2 与 > 结合代表错误重定向,而 1 则代表错误重定向到一个文件 1,而不代表标准输出;换成 2>&1,& 与 1 结合就代表标准输出了,就变成错误重定向到标准输出。

参考