一、课堂内容
1.fork()函数
2.运用fork()函数创建进程
二、fork()函数
1.代码实现
#include <unistd.h>
pid_t fork(void);
2.功能
创建进程;函数执行后,系统会创建一个与原进程几乎相同的进程,之后父子进程都继续执行。
3.返回值说明
- 成功:返回两个值,子进程创建成功后,原程序会被复制,就有了两个fork函数。父进程的fork函数会返回子进程的pid,子进程的fork函数会返回0
- 不成功:若子进程创建失败,原程序不会复制,父进程的fork函数返回-1
三、案例分析
1.使用fork创建进程,父子进程分别执行不同功能
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(){
pid_t tempPid;
tempPid = fork();
if(tempPid == -1){
perror("fork error");
}else if(tempPid > 0){//parent
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else{//child
printf("child process, pid = %d, ppid = %d\n", getpid(), getppid());
}//of if
printf("......finish......");
return 0;
}//of main
运行结果
【思考】多次运行代码会发现,child process后输出的ppid不等于parent process的pid,而等于1。
【原因】由于父进程与子进程的优先级相同,二者在争夺CPU的过程中,父进程可能会比子进程优先进入CPU运行,进而优先终止,在这种情况下,子进程就变成了“孤儿”进程,被init进程(即初始化进程)接收,这时候子进程输出的ppid就是init的pid,为1。
2.创建多个进程
方法:使用for循环
int i;
for(i = 0; i < 2; i ++){
tempPid = fork();
}//of for i
这个时候我们不难发现一个问题,即每一次循环,进程数量均会翻倍,这样会产生大量我们不需要的进程。如果只希望父进程可以创建新进程,则我们可以在for循环中添加一个判断:若当前进程不是父进程,则跳出循环。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(){
pid_t tempPid;
int i;
for(i = 0; i < 2; i ++){
if((tempPid = fork()) == 0){
break;
}//of if
}//of for i
if(tempPid == -1){
perror("fork error");
}else if(tempPid > 0){//parent
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else{//child
printf("I am child process = %d, pid = %d, ppid = %d\n", i + 1, getpid(), getppid());
}//of if
printf("......finish......");
return 0;
}//of main
运行结果
【思考】多次运行后我们发现,有时子进程的编号不是递增的,并且终端提示符后面仍然有子进程信息打印,而命令提示符在最后一行的开头闪烁。
【原因】Shell命令提示符也是1个进程,它会和父子进程进程一起竞争CPU。
3.利用sleep函数,修改进程的执行顺序,暂缓进程执行
#include <stdio.h>
#include <stdlib.h>
#include <unid.h>
int main(){
pid_t tempPid;
int i;
for(i = 0; i < 2; i ++){
if((tempPid = fork()) == 0){
break;
}//of if
}//of for i
if(tempPid == -1){
perror("fork error");
}else if(tempPid > 0){//parent
sleep(2);
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else{//child
sleep(i);
printf("I am child process = %d, pid = %d, ppid = %d\n", i + 1, getpid(), getppid());
}//of if
printf("......finish......");
return 0;
}//of main
运行结果
四、课后总结
学习了新的函数——fork(),知道了如何利用已有进程创建新的进程。在课堂上老师问到child process后输出的ppid不等于parent process的pid,而等于1的时候,我还愣了半天,还是在老师的点拨下才想到进程公平竞争CPU这一点。