要进行 bash 的 job control 必须要注意到的限制是:
- 这些工作所触发的程序必须来自於你 shell 的子程序(只管理自己的 bash);
- 前景:你可以控制与下达命令的这个环境称为前景的工作 (foreground);
- 背景:可以自行运行的工作,你无法使用 [ctrl]+c 终止他,可使用 bg/fg 呼叫该工作;
- 背景中『运行』的程序不能等待 terminal/shell 的输入(input)
- 直接将命令丢到背景中『运行』的 &
如同前面提到的,我们在只有一个 bash 的环境下,如果想要同时进行多个工作,那么可以将某些工作直接丢到背景环境当中,让我们可以继续操作前景的工作!那么如何将工作丢到背景中?最简单的方法就是利用『 & 』这个玩意儿了!举个简单的例子,我们要将 /etc/ 整个备份成为 /tmp/etc.tar.gz 且不想要等待,那么可以这样做:
[root@www ~]# tar -zpcf /tmp/etc.tar.gz /etc & [1] 8400 <== [job number] PID [root@www ~]# tar: Removing leading `/' from member names # 在中括号内的号码为工作号码 (job number),该号码与 bash 的控制有关。 # 后续的 8400 则是这个工作在系统中的 PID。至於后续出现的数据是 tar 运行的数据流, # 由於我们没有加上数据流重导向,所以会影响画面!不过不会影响前景的操作喔!
- 将『目前』的工作丢到背景中『暂停』:[ctrl]-z
想个情况:如果我正在使用 vi ,却发现我有个文件不知道放在哪里,需要到 bash环境下进行搜寻,此时是否要结束 vi 呢?呵呵!当然不需要啊!只要暂时将 vi 给他丢到背景当中等待即可。例如以下的案例:
[root@www ~]# vi ~/.bashrc # 在 vi 的一般模式下,按下 [ctrl]-z 这两个按键 [1]+ Stopped vim ~/.bashrc [root@www ~]# <==顺利取得了前景的操控权! [root@www ~]# find / -print ....(输出省略).... # 此时萤幕会非常的忙碌!因为萤幕上会显示所有的档名。请按下 [ctrl]-z 暂停 [2]+ Stopped find / -print |
- 观察目前的背景工作状态: jobs
[root@www ~]# jobs [-lrs] 选项与参数: -l :除了列出 job number 与命令串之外,同时列出 PID 的号码; -r :仅列出正在背景 run 的工作; -s :仅列出正在背景当中暂停 (stop) 的工作。 范例一:观察目前的 bash 当中,所有的工作,与对应的 PID [root@www ~]# jobs -l [1]- 10314 Stopped vim ~/.bashrc [2]+ 10833 Stopped find / -print |
如果想要知道目前有多少的工作在背景当中,就用 jobs 这个命令吧!一般来说,直接下达 jobs 即可!不过,如果你还想要知道该 job number 的 PID 号码,可以加上 -l 这个参数啦!在输出的资讯当中,例如上表,仔细看到那个 + - 号喔!那个 + 代表默认的取用工作。所以说:『目前我有两个工作在背景当中,两个工作都是暂停的,而如果我仅输入 fg 时,那么那个 [2] 会被拿到前景当中来处理』!
其实 + 代表最近被放到背景的工作号码, - 代表最近最后第二个被放置到背景中的工作号码。而超过最后第三个以后的工作,就不会有 +/- 符号存在了!- 将背景工作拿到前景来处理:fg
刚刚提到的都是将工作丢到背景当中去运行的,那么有没有可以将背景工作拿到前景来处理的?有啊!就是那个 fg (foreground) 啦!举例来说,我们想要将上头范例当中的工作拿出来处理时:
[root@www ~]# fg %jobnumber 选项与参数: %jobnumber :jobnumber 为工作号码(数字)。注意,那个 % 是可有可无的! 范例一:先以 jobs 观察工作,再将工作取出: [root@www ~]# jobs [1]- 10314 Stopped vim ~/.bashrc [2]+ 10833 Stopped find / -print [root@www ~]# fg <==默认取出那个 + 的工作,亦即 [2]。立即按下[ctrl]-z [root@www ~]# fg %1 <==直接规定取出的那个工作号码!再按下[ctrl]-z [root@www ~]# jobs [1]+ Stopped vim ~/.bashrc [2]- Stopped find / -print |
让工作在背景下的状态变成运行中: bg
离线管理问题
要注意的是,我们在工作管理当中提到的『背景』指的是在终端机模式下可以避免 [crtl]-c 中断的一个情境,并不是放到系统的背景去喔!所以,工作管理的背景依旧与终端机有关啦!在这样的情况下,如果你是以远程连线方式连接到你的 Linux 主机,并且将工作以 & 的方式放到背景去,请问,在工作尚未结束的情况下你离线了,该工作还会继续进行吗?答案是『否』!不会继续进行,而是会被中断掉。
那怎么办?如果我的工作需要进行一大段时间,我又不能放置在背景底下,那该如何处理呢?首先,你可以参考前一章的 at 来处理即可!因为 at 是将工作放置到系统背景,而与终端机无关。如果不想要使用 at 的话,那你也可以尝试使用 nohup 这个命令来处理喔!这个 nohup 可以让你在离线或注销系统后,还能够让工作继续进行。他的语法有点像这样:
[root@www ~]# nohup [命令与参数] <==在终端机前景中工作 [root@www ~]# nohup [命令与参数] & <==在终端机背景中工作 |
有够好简单的命令吧!上述命令需要注意的是, nohup 并不支持 bash 内建的命令,因此你的命令必须要是外部命令才行。我们来尝试玩一下底下的任务吧!
# 1. 先编辑一支会『睡著 500 秒』的程序: [root@www ~]# vim sleep500.sh #!/bin/bash /bin/sleep 500s /bin/echo "I have slept 500 seconds." # 2. 丢到背景中去运行,并且立刻注销系统: [root@www ~]# chmod a+x sleep500.sh [root@www ~]# nohup ./sleep500.sh & [1] 5074 [root@www ~]# nohup: appending output to ‘nohup.out’ <==会告知这个信息! [root@www ~]# exit |
如果你再次登陆的话,再使用 pstree 去查阅你的程序,会发现 sleep500.sh 还在运行中喔!并不会被中断掉!这样了解意思了吗?由於我们的程序最后会输出一个信息,但是 nohup 与终端机其实无关了,因此这个信息的输出就会被导向『 ~/nohup.out 』,所以你才会看到上述命令中,当你输入 nohup 后,会出现那个提示信息罗。
如果你想要让在背景的工作在你注销后还能够继续的运行,那么使用 nohup 搭配 & 是不错的运行情境喔!可以参考看看!