实验二:进程的控制
声明:.c文件里的代码虽不是本人亲自所写,但都在原来的基础上进行了修改,侵权必删
实验目的
通过进程的创建、撤消和运行加深对进程概念和进程并发执行的理解,明确进程与程序之间的区别。
实验内容
-
了解系统调用fork()、execvp()和wait()的功能和实现过程。
1.系统调用fork允许一个进程(父进程)创建新进程(子进程) 会从主程序中复制出一个新的程序,如果folk返回0就是子程序,否则那就是还是当前的程序。 2.系统调用wait 父进程用来获取子进程状态.在主程序中你可以自己决定要等待子程序返回才继续运算这样保持同步还是异步的不等子程序的结果就继续往下运行 3. execvp()函数用来执行指明的程序
-
编写一段程序,使用系统调用fork()来创建两个子进程,并由父进程重复显示字符串“parent:”和自己的标识数,而子进程则重复显示字符串“child:”和自己的标识数。
a——建立chap2-01.c 并打开文件touch chap2-01.c ls//查看是否建立成功
b——编写程序打开.c文件就不演示了
gedit chap2-01.c
#include<stdio.h> #include<stdlib.h> #include<sys/types.h> #include<sys/wait.h> #include<fcntl.h> #include<unistd.h> int main(){ if(fork==0){ printf("child ID1:%d\n",getpid()); }else{ wait(NULL); printf("child ID:%d\n",getpid()); if(fork==0){ printf("child ID2:%d\n",getpid()); }else{ wait(NULL); printf("child ID:%d\n",getpid()); } } return 0; }
c——运行程序
gcc chap2-01.c ./a.out
-
编写一段程序,使用系统调用fork()来创建一个子进程。子进程通过系统调用execvp()更换自己的执行代码,新的代码显示“new program.”。而父进程则调用wait()等待子进程结束,并在子进程结束后显示子进程的标识符,然后正常结束。
(就不截图啦)
a——建立chap2-02.c文件并打开文件touch chap2-02.c ls gedit chap2-02.c
b——编写程序
#include<stdio.h> #include<stdlib.h> #include<sys/types.h> #include<sys/wait.h> #include<unistd.h> int main(){ int p; char *a[]={"ls","-l","/etc/passwd",0}; p=fork(); if(p==0){ printf("new program!\n"); execvp("ls",a); exit(0); }else{ printf("parent=%d\n",getpid()); waitpid(p,NULL,0); printf("child=%d\n",getpid()); } return 0; }
c——运行程序
gcc chap2-02.c ./a.out
思考
(1)系统调用fork()是如何创建进程的?
(1)申请空白PCB (2)为新进程分配资源
(3)初始化进程控制块 (4)将新进程插入就绪队列
(2)当首次将CPU调度给子进程时,其入口在哪里?
fork系统调用创建的子进程继承了原进程的context,也就是说fork调用成功后,子进程与父进程并发执行相同的代码。
但由于子进程也继承了父进程的程序指针,所以子进程是从fork()后的语句开始执行(也就是新进程调用的入口)。
另外fork在子进程和父进程中的返回值是不同的。在父进程中返回子进程的PID,而在子进程中返回0。所以可以在程序中检查PID的值,使父进程和子进程执行不同的分支。
(3)系统调用exec()是如何更换进程的可执行代码的?
一个库函数通过调用execve来调用内核服务来执行指明的程序
(4)系统调用exit()是如何终止一个进程的?
execvp()会从PATH 环境变量所指的目录中查找符合参数file 的文件名,找到后便执行该文件,然后将第二个参数argv传给该欲执行的文件。
(5)对一个应用,如果多个进程的并发执行来实现,与单个进程
使用多个进程并发执行,则说明有许多个进程同时处理应用,比单个进程单独处理的效率要高,但是相对的并发执行消耗的资源页更多,要求也更高。