在Linux中,利用Shell的作业控制是比较常用的操作,在这一节中我们将探究作业控制相关的操作。为了方便我们查看区分不同的进行,我们编写如下程序,其功能是每间隔2秒输出一次自己的编号。
1 /* 2 ** Test puting a running program into backgound 3 */ 4 #include <stdio.h> 5 #include <stdlib.h> 6 7 int main(int argc, char* argv[]) 8 { 9 if (argc != 2){ 10 printf("Usage : bgtest number\n"); 11 return -1; 12 } 13 14 while (1){ 15 printf("Task [%d] Wait 2 seconds.\n", atoi(argv[1])); 16 sleep(2); 17 } 18 }
1 将进程放入后台
在shell编程环境中,可以使用组合键CTRL+Z将前台正在运行的程序放入后台,但是程序会暂停执行。示例如下:
xiaomanon@xiaomanon:~/Documents/Shell$ ./bgtest 2000 Task [2000] Wait 2 seconds. ^Z [1]+ Stopped ./bgtest 2000 xiaomanon@xiaomanon:~/Documents/Shell$ ./bgtest 2001 Task [2001] Wait 2 seconds. ^Z [2]+ Stopped ./bgtest 2001 xiaomanon@xiaomanon:~/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命令,用法如下:
xiaomanon@xiaomanon:~/Documents/Shell$ jobs [1] Stopped ./bgtest 2000 [2]- Stopped ./bgtest 2001 [3]+ Stopped ./bgtest 2002
该命令输出有3列,第一列为任务编号,第二列为任务的状态,第三列为程序名。我们可以看到,所有的进程被放入到后台以后全部都暂停了。
3 后台运行程序
如何让后台被暂停的程序重新运行呢?我们可以使用命令“bg 任务编号”来让程序在后台运行,如下所示就是让任务编号为1的程序重新运行。
xiaomanon@xiaomanon:~/Documents/Shell$ bg 1 [1] ./bgtest 2000 & xiaomanon@xiaomanon:~/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 任务编号”,如下所示:
xiaomanon@xiaomanon:~/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终止前台运行的进程。
xiaomanon@xiaomanon:~/Documents/Shell$ fg 3 ./bgtest 2002 Task [2002] Wait 2 seconds. ^C xiaomanon@xiaomanon:~/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并没有被杀死。
xiaomanon@xiaomanon:~/Documents/Shell$ kill 10687 xiaomanon@xiaomanon:~/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来杀死该前台进程。