zoukankan      html  css  js  c++  java
  • 探究wait与waitpid之间的那些事

    首先先说说进程

    1.进程产生
    操作运行了一个程序,至少会产生一个进程

    问题:进程与程序的区别?
    回答:程序就是一个编译好的可执行文件,而进程是操作系统为了执行一个程序而创建的,进程是程序的一次执行过程

    二 进程相关的命令

    1.ps -ef
    查看进程的PID和PPID

    2.ps aux
    查看进程状态 R:运行态 S:可中断的等待态 D:不能中断的等待态 T:停止态 Z:僵尸态(PID资源未释放)

    3.kill
    <1>kill -l 查看系统中信号
    <2>发送信号到指定的进程
    kill -信号 PID
    例如:
    kill -SIGSTOP 5207

    4.killall
    给同名的进程发信号

    killall -信号 进程名

    5.查看系统允许的最大进程个数
    cat /proc/sys/kernel/pid_max -> 32768

    三 创建子进程

    pid_t fork(void);
    功能:创建子进程
    返回值:
    成功给父进程返回子进程的PID,给进程返回0,失败返回-1

    思考:如何创建子进程?
    回答:复制父进程

    问题:为什么创建子进程?
    回答:创建子进程的目的是让子进程执行独立任务

    问题:子进程和父进程会不会相互影响
    回答:不会,父子进程的地址空间是独立

    四 父进程回收僵尸态子进程

    首先应该知道的概念:

    僵尸进程:进程结束,但是父进程没有对它进行收尸处理,即父进程没有对它做资源的回收处理,在内核进程树上还占用一个节点
    孤儿进程:父亲进程结束,子进程就会变成孤儿进程,孤儿进程会自动被init进程收养

    pid_t wait(int *status);
    功能:等待僵尸态子进程
    参数:
    @status 获取僵尸态子进程状态
    返回值:
    成功返回僵尸态子进程PID,失败返回-1(没有子进程)

    常用用法: wait(NULL);

    pid_t waitpid(pid_t pid, int *status, int options);
    参数:
    @pid -1:等待任意一个子进程退出 子进程的PID:等待指定PID子进程
    @status 获取子进程退出的状态
    @options 0:阻塞的方式调用 WNOHANG:非阻塞方式调用
    返回值:
    成功返回僵尸态子进程PID,失败返回-1, 如果没有子进程处于僵尸态,且非阻塞方式调用,此时立即返回,返回值为0

    waitpid(-1,NULL,0); => wait(NULL);
    waitpid(pid,NULL,0);

    waitpid(-1,NULL,WNOHANG);//常用的用法

    waitpid(pid,NULL,WNOHANG);

    五 结束一个进程

    return : 在main函数使用return,用来结束进程
    _exit() : 用来结束一个进程
    exit() : 用来结束一个进程,刷新缓存

    exit(EXIT_SUCCESS) 成功
    exit(EXIT_FAILURE) 失败

    六:实例如下:

    int main(int argc, const char *argv[])
    {
      int ret;
      int status;
      nt child_pid1;
      int child_pid2;

      child_pid1 = fork();
      if(child_pid1 < 0){
        perror("Fail to fork");
        return -1;
      }

      if(child_pid1 == 0){
        printf("hello word! ");
        return 0x18;
      }

        if(child_pid1 > 0){

        child_pid2 = fork();
      if(child_pid2 < 0){
        perror("Fail to fork");
        return -1;
      }

      if(child_pid2 == 0){
      while(1)
      ;
      }

      if(child_pid2 > 0){
      while(1)
      {
        printf("wait ... ");
        ret = wait(&status);

        printf("ret : %d ",ret);

        if(ret < 0){
        printf("Fail to wait : %s ",strerror(errno));
        return -1;
         }

      if(WIFEXITED(status)){
        printf("The child process normally exit , return : %#x ",WEXITSTATUS(status));
      }else if(WIFSIGNALED(status)){
        printf("The Child process terminated by signal : %d ",WTERMSIG(status));
         }

           }
         }  
      }

     return 0;
    }

    这个程序主要功能为,创建两个子进程,父进程分别等待两个子进程退出,一个为正常退出,一个通过发送信号使其退出

     测试结果如下

    编译执行:

    可以看到子进程1正常退出,用WEXITSTATUS(status)获取了她的返回码

    而子进程2一直在轮询,然后,我们查看后台进程

    执行 ps aux|grep a.out

    可以看到其中一个子进程的确已经退出,然后我们发送信号让另一个子进程退出(你能确定那个是子进程吗?)

    kill -SIGKILL 10611  (当然除了SIGKILL ,还可以被其他信号中断(具体执行kill -l),为什么?)

    结果如下:

    可知进程2已经被信号终止,并且通过WTERMSIG获取了是那个信号

    既然wait(&status),可以等待子进程退出,如果换成waitpid呢,请看下面的改造:

    测试如下:

    可以看出waitpaid(-1,&status,WNOHANGE),会去检测,当前有没有处于僵尸态的子进程,如果没有,立即返回,返回值为0

    同理发送信号回终止子进程2

     其他用法上面已经说过,就不举例说明!

  • 相关阅读:
    【NOIP2016提高组】 Day1 T3 换教室
    【NOIP2016 Day1 T2】天天爱跑步
    web@前端--html,css,javascript简介、第一个页面(常用标签简介)
    MySQL--视图view、触发器trigger、事务(start transaction)、存储过程(特殊的数据逻辑处理函数)、流程控制(if,case....)
    MySQL--pymysql模块
    MySQL--(了解)可能会用到的内置函数
    MySQL--详细查询操作(单表记录查询、多表记录查询(连表查询)、子查询)
    MySQL--表操作(约束条件foreign key关联表 多对1,多对多,1对1)
    MySQL--表操作(innodb表字段数据类型、约束条件)、sql_mode操作
    MySQL基本语句、存储引擎
  • 原文地址:https://www.cnblogs.com/bwbfight/p/9267753.html
Copyright © 2011-2022 走看看