体验进程的生命周期

体验进程的生命周期

代码:

#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//运行调试

![外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传](https://img-home.csdnimg.cn/images/20230724024159.png?origin_url=E%3A%5Ctypora%5C%E4%BD%9C%E4%B8%9A%5C%E8%BF%9B%E7%A8%8B%5C2.PNG&pos_id=img-vCyUbhVE-
1696385966980)

![外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传](https://img-home.csdnimg.cn/images/20230724024159.png?origin_url=E%3A%5Ctypora%5C%E4%BD%9C%E4%B8%9A%5C%E8%BF%9B%E7%A8%8B%5C3.PNG&pos_id=img-b3a1X8wr-
1696385966981)

![外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传](https://img-home.csdnimg.cn/images/20230724024159.png?origin_url=E%3A%5Ctypora%5C%E4%BD%9C%E4%B8%9A%5C%E8%BF%9B%E7%A8%8B%5C4.PNG&pos_id=img-QEUrU4uq-
1696385966982)

![外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传](https://img-home.csdnimg.cn/images/20230724024159.png?origin_url=E%3A%5Ctypora%5C%E4%BD%9C%E4%B8%9A%5C%E8%BF%9B%E7%A8%8B%5C6.PNG&pos_id=img-3vDuhYny-
1696385966983)

##### ![外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传](https://img-home.csdnimg.cn/images/20230724024159.png?origin_url=E%3A%5Ctypora%5C%E4%BD%9C%E4%B8%9A%5C%E8%BF%9B%E7%A8%8B%5C5.PNG&pos_id=img-iDrsd0si-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函数族其他函数没有了解全面,后续会继续学习相关内容。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值