在Linux中,利用Shell的作业控制是比较常用的操作,在这一节中我们将探究作业控制相关的操作。为了方便我们查看区分不同的进行,我们编写如下程序,其功能是每间隔2秒输出一次自己的编号。
1 /*
2 ** Test puting a running program into backgound3 */
4 #include
5 #include
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将前台正在运行的程序放入后台,但是程序会暂停执行。示例如下:
linuxidc@linuxidc:~/Documents/Shell$ ./bgtest 2000Task [2000] Wait 2seconds.^Z
[1]+ Stopped ./bgtest 2000linuxidc@linuxidc:~/Documents/Shell$ ./bgtest 2001Task [2001] Wait 2seconds.^Z
[2]+ Stopped ./bgtest 2001linuxidc@linuxidc:~/Documents/Shell$ ./bgtest 2002Task [2002] Wait 2seconds.^Z
[3]+ Stopped ./bgtest 2002
如上所示,我们运行了3个进程,并且给每个进程指定一个不同的编号,如2000、 2001、 2002,借此我们可以知道当前哪一个进程在运行。当我们按下Ctrl+Z组合键后,进程会被放到后台并且暂停,同时我们可以看到输出了一行字符串,类似于“[1]+ Stopped ./bgtest 2000”,要注意中括号[]中的数字就是系统自动生成的后台任务编号。
2 查看后台程序
在前面,我们已经将3个进程放到后台了,但是我们如何查看呢?可以使用jobs命令,用法如下:
linuxidc@linuxidc:~/Documents/Shell$ jobs
[1] Stopped ./bgtest 2000[2]- Stopped ./bgtest 2001[3]+ Stopped ./bgtest 2002
该命令输出有3列,第一列为任务编号,第二列为任务的状态,第三列为程序名。我们可以看到,所有的进程被放入到后台以后全部都暂停了。
3 后台运行程序
如何让后台被暂停的程序重新运行呢?我们可以使用命令“bg 任务编号”来让程序在后台运行,如下所示就是让任务编号为1的程序重新运行。
linuxidc@linuxidc:~/Documents/Shell$ bg 1[1] ./bgtest 2000 &linuxidc@linuxidc:~/Documents/Shell$ Task [2000] Wait 2seconds.
Task [2000] Wait 2seconds.
joTask [2000] Wait 2seconds.
bs
[1] Running ./bgtest 2000 &[2]- Stopped ./bgtest 2001[3]+ Stopped ./bgtest 2002
我们可以看到,程序会继续输出字符串,但是,真个并不影响我们接着执行其他命令,比如我们这是继续使用jobs命令,可以查看到后台运行的程序的状态,也看到任务1的状态改为了“Running”。
4 让程序前台运行
当然,我们也可以让被放到后台的程序重新回到前台运行,可以执行“fg 任务编号”,如下所示:
linuxidc@linuxidc:~/Documents/Shell$ fg 3./bgtest 2002Task [2002] Wait 2seconds.
Task [2002] Wait 2seconds.
Task [2002] Wait 2seconds.
jobTask [2002] Wait 2seconds.
s
Task [2002] Wait 2seconds.
jos
Task [2002] Wait 2seconds.
Task [2002] Wait 2seconds.
Task [2002] Wait 2seconds.^Z
[3]+ Stopped ./bgtest 2002
可以看到,程序在前台运行时,如果我们输入其他命令是不能响应的,整个前台被当前运行的任务3独占。我们必须将其重新放入到后台才能够输入其他命令。
5 杀死前台进程
杀死前台进程比较容易,我们可以直接使用组合键Ctrl+C终止前台运行的进程。
linuxidc@linuxidc:~/Documents/Shell$ fg 3./bgtest 2002Task [2002] Wait 2seconds.^C
linuxidc@linuxidc:~/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 COMMAND1000 9301 0.0 0.6 9868 6332 pts/0 Ss 09:47 0:01 -bash1000 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并没有被杀死。
linuxidc@linuxidc:~/Documents/Shell$ kill 10687linuxidc@linuxidc:~/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 COMMAND1000 9301 0.0 0.6 9868 6332 pts/0 Ss 09:47 0:01 -bash1000 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来杀死该前台进程。