进程控制
实验目的
- 加深对进程概念的理解,明确进程和程序的区别。
- 掌握Linux系统中的进程创建,管理和删除等操作。
- 熟悉使用Linux下的命令和工具,如man, find, grep, whereis, ps, pgrep, kill, ptree, top, vim, gcc,gdb, 管道|等。
基础知识
- 进程的创建
Linux中,载入内存并执行程序映像的操作与创建一个新进程的操作是分离的。将程序映像载入内存,并开始运行它,这个过程称为运行一个新的程序,相应的系统调用称为exec系统调用。而创建一个新的进程的系统调用是fork系统调用 - exec系统调用
#include <unistd.h>
int execl (const char *path, const char *arg,...);
execl()将path所指路径的映像载入内存,arg 是它的第一个参数。参数可变长。参数列表必须以NULL结尾。
通常execl()不会返回。成功的调用会以跳到新的程序入口点作为结束。发生错误时,execl()返回-1,并设置errno值。
例 编辑/home/kidd/hooks.txt:
int ret;
ret = execl (”/bin/vi”, ”vi”,”/home/kidd/hooks.txt”, NULL);
if (ret == -1)
perror (”execl”);
- fork系统调用
#include <sys/types.h>
#include <unistd.h>
pid_t fork (void);
成功调用fork()会创建一个新的进程,它与调用fork()的进程大致相同。发生错误时,fork()返回-1,并设置errno值。
例:
pid_t pid;
pid = fork ();
if (pid > 0)
printf (”I am the parent of pid=%d!\n”, pid);
else if (!pid)
printf (”I am the baby!\n”);
else if (pid == -1)
perror (”fork”);
- 终止进程
exit()系统调用:
#include <stdlib.h>
void exit (int status);
- 进程挂起
pause() 系统调用:
int pause( void );
函数pause会把进程挂起,直到接收到信号。在信号接收后,进程会从pause函数中退出,继续运行。
- wait(等待子进程中断或结束)
#include<sys/types.h>
#include<sys/wait.h>
pid_t wait (int * status);
wait()会暂时停止目前进程的执行,直到有信号来到或子进程结束。
如果在调用 wait()时子进程已经结束,则wait()会立即返回子进程结束状态值。
子进程的结束状态值会由参数status返回,而子进程的进程识别码也会一起返回。
如果不在意结束状态值,则参数status可以设成 NULL。
实验内容
第一题
题目:打开一个vi进程。通过ps命令以及选择合适的参数,只显示名字为vi的进程。寻找vi进程的父进程,直到init进程为止。记录过程中所有进程的ID和父进程ID。将得到的进程树和由pstree命令的得到的进程树进行比较。
过程:
打开一个vi进程(输入vi回车)
另打开一个终端用于ps命令
用ps -A 命令查看vi进程号是4697
ps -p4697 查看vi运行情况
ps -lax显示所有进程的信息并查找vi的父进程直到init(PID为1)
UID:用户号
PID:进程的进程号
PPID:进程的父进程号,
PRI:内核调度优先级
NI:是进程优先级, 为缺省值0;
VSZ:是总虚拟内存大小, 以byte计;
RSS:是进程使用的总物理内存数,以 Kbytes计;
STAT:是进程状态, 属于静止状态;
TTY:是终端的次要装置号码;
TIME:为使用cpu的时间;
4697 -> 4686 -> 4681 -> 1499 -> 1488 -> 1213 -> 1(init)
pstree -p
两种方式结果一样
第二题
编写程序,首先使用fork系统调用,创建子进程。在父进程中继续执行空循环操作;在子进程中调用exec打开vi编辑器。然后在另外一个终端中,通过ps –Al命令、ps aux或者top等命令,查看vi进程及其父进程的运行状态,理解每个参数所表达的意义。选择合适的命令参数,对所有进程按照cpu占用率排序
程序如下:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(int argc, char **argv)
{ int pid =fork();
if(pid>0){
int ret;
ret=execlp("vi","",NULL);
if(ret==-1){
printf("execl error\n");
exit(-1);}}
else {printf("fork error\n");
exit(-1);}
return 0;}
使用ps -Al命令
•F代表程序的旗标(flag),4代表使用者为 superuser,1代表user;
•S代表这个程序的状态 (STAT);
•UID代表执行者身份;
•PID代表进程的ID号
•PPID代表父进程的ID;
•C代表使用的CPU资源百分比;
•PRI代表进程的执行优先权(Priority的简写),其值越小越早被执行;
•NI代表进程的nice值,其表示进程可被执行的优先级的修正数值;
•ADDR代表内核函数;
•SZ表示占用内存的大小;
•WCHAN表示这个程序是否正在运作当中,其中’ -’表示正在运作;
•TTY表示登入者的终端机位置;
•TIME表示使用掉的 CPU 时间;
•CMD 表示下达指令的名称;
使用命令ps aux查看内存中运行的程序,查阅资料可知:
• USER:该进程属于的使用者账号;
• PID:该进程的进程ID号;
• CPU:该进程使用掉的CPU资源百分比;
• MEM:该进程所占用的物理内存百分比;
• VSZ:该进程使用掉的虚拟内存量 (Kbytes)
• RSS :该进程占用的固定的内存量 (Kbytes)
• TTY :该进程是在那个终端机上面运作,若与终端机无关,则显示 ?,另外, tty1-tty6 是本机上面的登入者程序,若为 pts/0 等等的,则表示为由网络连接进主机的程序。
• STAT:该程序目前的状态,主要的状态有:
R :该程序目前正在运作,或者是可被运作;
S :该程序目前正在睡眠当中;
T :该程序目前正在侦测或者是停止了;
Z :该程序应该已经终止,但是其父程序却无法正常的终止他,造成 zombie(疆尸) 程序的状态;
• START:该进程被触发启动的时间;
• TIME :该进程实际使用 CPU 运作的时间。
• COMMAND:该进程所属的指令
Top指令(动态变化)
将所有进程按照cpu占有率排名,可见死循环程序fork_exec已经几乎完全占有了CPU:
第三题
使用fork系统调用,创建如下进程树,并使每个进程输出自己的ID和父进程的ID。观察进程的执行顺序和运行状态的变化。
#include <stdio.h>
#include <unistd.h>
int main(){
pid_t p1;
printf("p1 pid: %d, ppid: %d\n", getpid(),getppid());
if(p1==0)
{
pid_t p3;
p3 = fork();
if(p3 == 0)
printf("p3 pid: %d, ppid: %d\n", getpid(),getppid());
else if(p3 > 0){
pid_t p2;
p2 = fork();
if(p2==0){
printf("p2 pid: %d, ppid: %d\n", getpid(),getppid());
pid_t p4;
p4 = fork();
if(p4 == 0){
printf("p4 pid: %d, ppid: %d\n", getpid(),getppid());}
else if(p4 >0){
pid_t p5;
p5 = fork();
if(p5==0)
printf("p5 pid: %d, ppid: %d\n", getpid(),getppid());}}}}
sleep(1);
}
第四题
修改上述进程树中的进程,使得所有进程都循环输出自己的ID和父进程的ID。然后终止p2进程(分别采用kill -9 、自己正常退出exit()、段错误退出),观察p1、p3、p4、p5进程的运行状态和其他相关参数有何改变。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main()
{
pid_t p1;
if(p1==0){
pid_t p3;
p3 = fork();
if(p3 == 0){
int i;
for(i=0;i<10;i++){
printf("p3 pid: %d, ppid: %d\n", getpid(),getppid());
sleep(1);}
return 0;}
else if(p3 > 0){
pid_t p2;
p2 = fork();
if(p2==0){
pid_t p4;
p4 = fork();
if(p4 == 0){
int i;
for(i=0;i<10;i++){
printf("p4 pid: %d, ppid: %d\n", getpid(),getppid());
sleep(1);}
return 0;}
else if(p4 >0){
pid_t p5;
p5 = fork();
if(p5==0){
int i;
for(i=0;i<10;i++){
printf("p5 pid: %d, ppid: %d\n", getpid(),getppid());
sleep(1);}
return 0;}
else{//p2
int i;
for(i=0;i<10;i++){
if(i==5)
exit(1);
/*段错误
if(i==5)
{
int *p=NULL;
*p=0;
}*/
printf("p2 pid: %d, ppid: %d\n", getpid(),getppid());
sleep(1);}
return 0;}}}}
int i;
for(i=0;i<10;i++)
{printf("p1 pid: %d, ppid: %d\n", getpid(),getppid());
sleep(1);}
return 0;}
sleep(1);}
设循环十次
GitHub代码链接:
https://github.com/notdz56/16281023-/tree/master/操作系统第二次实验代码