体验进程的生命周期
代码:
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
int main()
{
int status;//保存被收集进程退出时的一些状态
pid_t pid=fork();//创建新进程
pid_t npid=getpid();
if(pid<0)//进程有误
{
perror("fork error\n");
return 1;
}
else if(pid==0)//子进程
{
printf("child process is %d\n",npid);
//exec系统调用替换子进程
execlp("/bin/ls","ls","-l",NULL);//执行ls命令
perror("execlp error");
exit(0);
}else
{
//父进程
printf("father process is %d\n",npid);
//等待子进程结束
wait(&status);
if(WIFEXITED(status))//子进程是否正常退出
{
printf("child process exited with status code is %d\n",WIFEXITED(status));
}else{
printf("child error");
}
sleep(4);//休眠4秒
exit(0);//退出进程
}
return 0;
}
调试结果:
gcc -c -g forktest.c
gcc forktest.o
gdb forktest.out
break 断点
info b //查看断点信息
r//运行调试
1696385966980)
1696385966981)
1696385966982)
1696385966983)
fork():
调用fork()的进程就是父进程,系统调用getpid()获取进程标识符,调用fork()之后,子进程就会被创建,其实就当一个进程调用时,就会产生与之一模一样的进程,即先创建父进程,将父进程代码复制到子进程,产生两个一模一样的进程。fork有三种不同的返回值即在父进程中返回子进程的进程ID,子进程中返回值为0,负值则出现错误。
wait():
原型为pid_t wait(int *status),status保存收集进程退出时的一些状态,主要用来判别子进程是否是正常退出,若设置为NULL则将此僵尸进程消灭掉,若不为NULL,则wait会将子进程退出时的状态取出并存入,存入信息被放到一个整数的不同二进制位中。
(1)WIFEXITED(status):status为整数,检查子进程是否正常退出,是则返回非零值。
(2)WEXITSTATUS(status):获取子进程返回值,返回结果为非零值。
execlp():
exec将进程创建与加载一个新进程映象分离,可以更方便对两种操作进行管理。其实就是用exec函数可以把当前进程替换为一个新进程,且新进程与原进程有相同的PID,替换进程的过程中是不会创建新进程的,因为进程替换只是将该进程的数据替换为指定的可执行程序。而进程PCB没有改变,所以不是新的进程,进程替换后不会发生进程pid,exec系列函数execl、execlp、execle、execv、execvp等等。
原型:int execlp(const char *file, const char *arg, …);采用execlp时它会在环境变量PATH当中查找命令。
file参数要启动程序的名称包括路径名;
arg参数启动程序所带的参数,一般第一个参数为要执行命令名,不是带路径且arg必须以NULL结束;
sleep():
显而易见,sleep 就是让程序稍稍休息一下,然后,再继续工作,而sleep的进程是不占据cpu时间的。
exit():
原型:void exit(int status);status为整数类型参数,其函数作用是无条件停止剩下的所有操作,清除各种数据结构,并终止本进程的运行。当调用exit()之后,进程终止之前要检查该进程打开了哪些文件,并把文件缓冲区中的内容写回文件,即资源等待父进程的wait进行收回。
心得体会
通过调试可得,没有创建新进程fork之前只有父进程在执行,其PID为8136,直到进fork系统调用,产生子进程pid为8247,当pid大于0时候我们向下执行父进程操作遇到wait()函数,用status保存收集子进程8247退出时的一些状态,而子进程8247中加入exec函数族里面的excelp()进行进程替换,将其替换成/bin/ls程序,打印当前目录下的文件信息,父进程等待子进程执行完毕,进入sleep阶段,暂停4秒,exit退出程序。本次实验自己动手让我对fork有了新的理解,对进程的创建、替换、阻塞、休眠以及退出程序有了更加深刻的认识,但对父进程阻塞以及exce函数族其他函数没有了解全面,后续会继续学习相关内容。