在Linux中,利用Shell的作业控制是比较常用的操作,在这一节中我们将探究作业控制相关的操作。为了方便我们查看区分不同的进行,我们编写如下程序,其功能是每间隔2秒输出一次自己的编号。
/*
** Test puting a running program into backgound
*/
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char* argv[])
{
if (argc != 2){
printf("Usage : bgtest number\n");
return -1;
}
while (1){
printf("Task [%d] Wait 2 seconds.\n", atoi(argv[1]));
sleep(2);
}
}
1.将进程放入后台
在shell编程环境中,可以使用组合键CTRL+Z将前台正在运行的程序放入后台,但是程序会暂停执行。示例如下:
linux@linux:~/Documents/Shell$ ./bgtest 2000
Task [2000] Wait 2 seconds.
^Z
[1]+ Stopped ./bgtest 2000
linux@linux:~/Documents/Shell$ ./bgtest 2001
Task [2001] Wait 2 seconds.
^Z
[2]+ Stopped ./bgtest 2001
linux@linux:~/Documents/Shell$ ./bgtest 2002
Task [2002] Wait 2 seconds.
^Z
[3]+ Stopped ./bgtest 2002
如上所示,我们运行了3个进程,并且给每个进程指定一个不同的编号,如2000、 2001、 2002,借此我们可以知道当前哪一个进程在运行。当我们按下Ctrl+Z组合键后,进程会被放到后台并且暂停,同时我们可以看到输出了一行字符串,类似于“[1]+ Stopped ./bgtest 2000”,要注意中括号[]中的数字就是系统自动生成的后台任务编号。
2.查看后台程序
在前面,我们已经将3个进程放到后台了,但是我们如何查看呢?可以使用jobs命令,用法如下:
linux@linux:~/Documents/Shell$ jobs
[1] Stopped ./bgtest 2000
[2]- Stopped ./bgtest 2001
[3]+ Stopped ./bgtest 2002
该命令输出有3列,第一列为任务编号,第二列为任务的状态,第三列为程序名。我们可以看到,所有的进程被放入到后台以后全部都暂停了。
3.后台运行程序
如何让后台被暂停的程序重新运行呢?我们可以使用命令“bg 任务编号”来让程序在后台运行,如下所示就是让任务编号为1的程序重新运行。
linux@linux:~/Documents/Shell$ bg 1
[1] ./bgtest 2000 &
linux@linux:~/Documents/Shell$ Task [2000] Wait 2 seconds.
Task [2000] Wait 2 seconds.
joTask [2000] Wait 2 seconds.
bs
[1] Running ./bgtest 2000 &
[2]- Stopped ./bgtest 2001
[3]+ Stopped ./bgtest 2002
我们可以看到,程序会继续输出字符串,但是,真个并不影响我们接着执行其他命令,比如我们这是继续使用jobs命令,可以查看到后台运行的程序的状态,也看到任务1的状态改为了“Running”。
4.让程序前台运行
当然,我们也可以让被放到后台的程序重新回到前台运行,可以执行“fg 任务编号”,如下所示:
linux@linux:~/Documents/Shell$ fg 3
./bgtest 2002
Task [2002] Wait 2 seconds.
Task [2002] Wait 2 seconds.
Task [2002] Wait 2 seconds.
jobTask [2002] Wait 2 seconds.
s
Task [2002] Wait 2 seconds.
jos
Task [2002] Wait 2 seconds.
Task [2002] Wait 2 seconds.
Task [2002] Wait 2 seconds.
^Z
[3]+ Stopped ./bgtest 2002
可以看到,程序在前台运行时,如果我们输入其他命令是不能响应的,整个前台被当前运行的任务3独占。我们必须将其重新放入到后台才能够输入其他命令。
5.杀死前台进程
杀死前台进程比较容易,我们可以直接使用组合键Ctrl+C终止前台运行的进程。
linux@linux:~/Documents/Shell$ fg 3
./bgtest 2002
Task [2002] Wait 2 seconds.
^C
linux@linux:~/Documents/Shell$ ps -u
Warning: bad ps syntax, perhaps a bogus '-'? See http://procps.sf.net/faq.html
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
1000 9301 0.0 0.6 9868 6332 pts/0 Ss 09:47 0:01 -bash
1000 10685 0.0 0.0 2008 284 pts/0 T 13:24 0:00 ./bgtest 2000
1000 10687 0.0 0.0 2008 280 pts/0 T 13:24 0:00 ./bgtest 2001
1000 10769 0.0 0.1 4944 1156 pts/0 R+ 13:43 0:00 ps -u
在我们使用命令“ps -u”查看当前用户进程时,可以发现任务3已经被杀死了。
6.杀死后台进程
我们知道,可以利用“kill 进程号”的方式来杀死正在运行的进程,但是对于暂停的进程能不能起作用呢?我们做了如下尝试,结果发现任务2并没有被杀死。
linux@linux:~/Documents/Shell$ kill 10687
linux@linux:~/Documents/Shell$ ps -u
Warning: bad ps syntax, perhaps a bogus '-'? See http://procps.sf.net/faq.html
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
1000 9301 0.0 0.6 9868 6332 pts/0 Ss 09:47 0:01 -bash
1000 10685 0.0 0.0 2008 284 pts/0 T 13:24 0:00 ./bgtest 2000
1000 10687 0.0 0.0 2008 280 pts/0 T 13:24 0:00 ./bgtest 2001
1000 10780 0.0 0.1 4944 1156 pts/0 R+ 13:45 0:00 ps -u
由此,我们只能够先使用fg命令将进程放到前台执行,然后再使用Ctrl+C来杀死该前台进程。