@ju1900
2017-08-18T13:55:48.000000Z
字数 2615
阅读 2347
linux
我们计算的程序都是周期很长的,通常要几个小时甚至一个星期。我们用的环境是用putty远程连接到日本Linux服务器。所以使程序在后台跑有以下三个好处:
方法有很多,这里主要列举两种。假如我们有程序pso.cpp,通过编译后产生可执行文件pso,我们要使pso在linux服务器后台执行。当客户端关机后重新登入服务器后继续查看本来在终端输出的运行结果。(假设操作都在当前目录下)
# ./pso > pso.file 2>&1 &
解释:将pso直接放在后台运行,并把终端输出存放在当前目录下的pso.file文件中。
当客户端关机后重新登陆服务器后,直接查看pso.file文件就可看执行结果(命令:#cat pso.file )。
# nohup ./pso > pso.file 2>&1 &
解释:nohup就是不挂起的意思,将pso直接放在后台运行,并把终端输出存放在当前目录下的pso.file文件中。当客户端关机后重新登陆服务器后,直接查看pso.file
文件就可看执行结果(命令:#cat pso.file )。
# jobs //查看任务,返回任务编号n和进程号
# bg [%]n //将编号为n的任务转后台运行
# fg [%]n //将编号为n的任务转前台运行
# ctrl+z //挂起当前任务
# ctrl+c //结束当前任务
在Linux中,如果要让进程在后台运行,一般情况下,我们在命令后面加上&即可,实际上,这样是将命令放入到一个作业队列中了:
$ ./test.sh &
[1] 17208
$ jobs -l
[1]+ 17208 Running ./test.sh &
对于已经在前台执行的命令,也可以重新放到后台执行,首先按ctrl+z暂停已经运行的进程,然后使用bg命令将停止的作业放到后台运行:
$ ./test.sh
[1]+ Stopped ./test.sh
$ bg 1
$ jobs -l
[1]+ 22794 Running ./test.sh &
但是如上方到后台执行的进程,其父进程还是当前终端shell的进程,而一旦父进程退出,则会发送hangup信号给所有子进程,子进程收到hangup以后也会退出。如果我们要在退出shell的时候继续运行进程,则需要使用setsid将将父进程设为init进程(进程号为1)
lte@ltepc42vm4:~$ ping 8.8.8.8
PING 8.8.8.8 (8.8.8.8) 56(84) bytes of data.
64 bytes from 8.8.8.8: icmp_seq=1 ttl=42 time=31.9 ms
^Z
[1]+ Stopped ping 8.8.8.8
lte@ltepc42vm4:~$ echo $$
7673
lte@ltepc42vm4:~$ !ps
ps -ef | grep ping
lte 7700 7673 0 13:54 pts/4 00:00:00 ping 8.8.8.8 # 父进程是7673
lte 7702 7673 0 13:54 pts/4 00:00:00 grep --color=auto ping
lte@ltepc42vm4:~$ setsid ping 8.8.8.8 > /dev/null
lte@ltepc42vm4:~$ echo $$
7773
lte@ltepc42vm4:~$ ps -ef | grep ping
lte 7800 1 0 13:58 ? 00:00:00 ping 8.8.8.8 # 父进程是1
lte 7802 7773 0 13:58 pts/2 00:00:00 grep --color=auto ping
上面的试验演示了使用nohup/setsid加上&使进程在后台运行,同时不受当前shell退出的影响。
那么对于已经在后台运行的进程,该怎么办呢?可以使用disown
命令:
lte@ltepc42vm4:~$ ping 8.8.8.8 > /dev/null &
[1] 7875
lte@ltepc42vm4:~$ !ps
ps -ef | grep ping
lte 7875 7852 0 14:06 pts/4 00:00:00 ping 8.8.8.8
lte 7877 7852 0 14:06 pts/4 00:00:00 grep --color=auto ping
lte@ltepc42vm4:~$ echo $$
7852
lte@ltepc42vm4:~$ jobs
[1]+ Running ping 8.8.8.8 > /dev/null &
lte@ltepc42vm4:~$ disown -h %1
lte@ltepc42vm4:~$ !ps
ps -ef | grep ping
lte 7875 7852 0 14:06 pts/4 00:00:00 ping 8.8.8.8
lte 7879 7852 0 14:07 pts/4 00:00:00 grep --color=auto ping
lte@ltepc42vm4:~$ exit # 退出后查看进程
lte@ltepc42vm4:~$ !ps
ps -ef | grep ping
lte 7875 1 0 14:06 ? 00:00:00 ping 8.8.8.8 # 这里的PID已经变成1, 不受shell退出影响
lte 7937 7918 0 14:07 pts/2 00:00:00 grep --color=auto ping
另外还有一种方法,即使将进程在一个subshell中执行,其实这和setsid异曲同工。方法很简单,将命令用括号() 括起来即可:
lte@ltepc42vm4:~$ (ping 8.8.8.8 > /dev/null &)
lte@ltepc42vm4:~$ !ps
ps -ef | grep ping
lte 7943 1 0 14:09 pts/2 00:00:00 ping 8.8.8.8
lte 7945 7918 0 14:09 pts/2 00:00:00 grep --color=auto ping
终止当前后台运行的程序
ps -ef | grep command
kill -9 pid
pkill command