说明

本文转载于https://beltxman.com/2524.html

使用SSH终端(比如putty,xshell)连接Linux服务器上运行比较耗时任务,可能需要几个小时甚至几天才能完成,这时候终端被占据,我们还要做其他的任务呢,所以需要将这一类耗时任务放到后台来跑,只需要记录好日志就行了,有下面这些常用方法来实现这个需求。

command &

在终端输入command &运行,这时候使用Ctrl + C中止命令将会失效,因为对&后台运行SIGINT信号免疫,程序会继续运行。但如果你直接关掉终端,进程将跟着中止。可见,使用&后台运行的进程会被SIGHUP信号中止,但是你如果用exit命令正常退出终端,程序将继续运行在后台不会中止。

说明⚠️:

(1)SIGINT信号:程序终止(interrupt)信号, 在用户键入INTR字符(通常是Ctrl-C)时发出,用于通知前台进程组终止进程

(2)SIGHUP信号:终端连接结束时发出, 通常是关闭终端, 通知同一session内的各个进程组

nohup command

标准输出会被默认追加到当前目录下的nohup.out中,如果当前目录的nohup.out文件不可写,输出重定向到$HOME/nohup.out文件中。

那么,我们很容易想到将这两个特性综合一下这样运行命令:

nohup command > /home/user/myfile.log 2>&1 &

这样将会在后台一直执行这个任务直到完成,并将程序标准输出(1)标准错误(2)都定向到/home/user/myfile.log文件中。

screen 工具

screen 可以理解为一个会话窗口管理工具,对于一般的使用,我们只需要会使用下面操作即可:

# 使用yum安装screen
yum install screen
# 创建一个名为test的会话窗口
screen -S test
# 暂离窗口
Ctrl+a d(即按住Ctrl,依次再按a,d)
# 查看存在的会话窗口
screen -ls
# 进入窗口
screen -r test
screen -r 进程ID
# 关闭窗口
exit
# 窗口切换
Ctrl+a c :在当前screen会话中创建窗口
Ctrl+a w :窗口列表
Ctrl+a n :下一个窗口
Ctrl+a p :上一个窗口
Ctrl+a 0-9 :在第0个窗口和第9个窗口之间切换

有了 screen 工具,我们就可以在终端上创建窗口,然后运行需要的命令,然后暂离,然后就可以继续做其他的事情了,当然,也可以用其他的终端连接服务器后使用screen -r命令连接需要的窗口,会发现程序一直在运行,退出终端也不会影响到那些暂离的终端上运行的程序。

测试

我们写一个测试脚本test.sh

#!/bin/sh
for ((i=1; i<1000; i++))
do
    d=`date '+%Y-%m-%d %H:%M:%S'`
    echo "$d 第 $i 次输出;"
    tt
    sleep 2s
done

只使用&后台运行:

[root@localhost test.cc]# ./test.sh > test.log 2>&1 &
[1] 15037
[root@localhost test.cc]# ps -aux | grep test.sh
root     15037  0.0  0.0 113180  1424 pts/0    S    16:10   0:00 /bin/sh ./test.sh
root     15045  0.0  0.0 112712   992 pts/0    S+   16:10   0:00 grep --color=auto test.sh

可以看到15037进程在运行;

如果直接关闭终端,日志记录就停止了,说明进程随着终端会话的结束而停止执行了;

但是,经过反复测试发现,如果使用exit命令正常退出终端,再连上终端后,使用&运行的程序继续在运行,也就是&命令跟nohup效果是一样的,我这里使用的测试环境是CentOS7.6。

使用nohup & 运行

[root@localhost test.cc]# nohup ./test.sh > test.log 2>&1 &
[1] 14349

测试结果,直接关闭终端或者使用exit命令退出终端,程序都会继续执行。

说明⚠️:

我在使用阿里云nas做测试时,使用 nohup & 这种方法执行命令后,直接关闭终端,程序会结束的!这点请大家注意~

使用screen运行

# 创建窗口
screen -S test
# 运行命令,这里可以不用后面的&,只是程序会把窗口占用
./test.sh > test.log 2>&1 &
# 暂离
Ctrl + a d

这时候可以直接关闭终端了,然后重新连接,发现程序依然在运行,如果要找到窗口连接:

# 存在的窗口
[root@localhost ~]# screen -ls
There is a screen on:
    18048.test  (Detached)
1 Socket in /var/run/screen/S-root.
# 进入窗口
[root@localhost ~]# screen -r test

进入之后会发现,窗口依然保持着暂离之前的状态,包括之前输入的程序和命令,当你在窗口中输入exit命令退出窗口,如果你前面命令后面没有使用&,程序将会中止,因为相对于test窗口来说,这是一个前台程序,窗口退出了程序自然就跟着中止了。

因为screen的暂离功能,可以实现暂离窗口中保持前台程序的继续运行,相对于真正的终端而言,就相当于后台运行了。