Linux系统编程之进程(三) 进程退出、子进程退出状态的收集、孤儿进程、僵尸进程

进程退出

进程退出分正常退出异常退出
正常退出有五种:

  1. main函数调用return
  2. 进程调用exit(),标准C库
  3. 进程调用_exit() 或者 _Exit(),属于系统调用
  4. 进程最后一个线程返回
  5. 最后一个先线程调用 pthread_exit
/*它要检查文件的打开情况,把文件缓冲区的内容写回文件,即“清理I/O缓冲”。*/
exit();//exit(0)正常退出;exit(1)异常退出
/*直接退出*/
_exit();//_exit(0)正常退出;_exit(1)异常退出
_Exit();//_Exit(0)正常退出;_Exit(1)异常退出

异常退出:

  1. 调用abort
  2. 当进程收集到某些信号时,如Ctrl + C
  3. 最后一个线程对取消(cancellation)请求做出
    不管进程如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。对上述任意一种终止情形、我们都希望终止进程能够通知其父进程它是如何终止的。对于三个终止函数(exit._exit和_Exit),实现这一点的方法是,将其退出状态((exit status)作为参数传送给函数。在异常终止情况下,内核(不是进程本身)产生一个指示其异常终止原因的终止状态(termination status)。在任意一种情况下,该终止进程的父进程都能用wait或waitpid函数取得其终止状态。

子进程退出状态的收集

子进程退出状态不被收集,就会变成僵尸进程(僵死进程)

int main()
{   
    int cnt =0;
        int fork_r=0;
        fork_r=vfork();
        if(fork_r!=0){//父进程
            while(1){
                    printf("this is the father process!\n");
                    sleep(2);
                    printf("cnt =%d\n",cnt);
            }
        }else{//子进程
            while(1){
                    printf("this is the child process!\n");
                    sleep(2);
                    if(cnt == 3){
                            exit(0);
                    }
                    cnt++;
            }
        }
}

收集子进程退出状态的API wait, waitpid, waitid
wait() 函数
头文件

#include <sys/types.h>
#include <sys/wait.h>

函数原型

pid_t wait(int *status);

status参数:是一个整型数指针
非空:子进程退出状态放在它所指向的地址中
空:不关心退出状态
返回值: 如果成功,返回终止子进程的进程ID;错误,返回-1

int main()
{
    int cnt=0;
    int fork_r=0;
    int status=0;
    fork_r=fork();
    if(fork_r!=0){//父进程
        wait(&status);//父进程会阻塞在这里,直到子进程结束
         printf("status=%d\n",WEXITSTATUS(status));
         while(1){
                 printf("this is the father process!,PID=%d\n",getpid());
                 sleep(2);
                 printf("cnt =%d\n",cnt);
         }
    }else{//子进程
         while(1){
                 printf("this is the child process!,PID=%d\n",getpid());
                 sleep(2);
                 if(cnt == 3){
                         exit(2);
                 }
                 cnt++;
         }
    }
}

孤儿进程

父进程不等待子进程退出,在子进程退出之前就已经执行结束,此时子进程叫做孤儿进程。避免出现过多的孤儿进程,inti进程(系统初始化进程,pid为1)会收留孤儿进程,成为孤儿进程的父进程

int main()
{
    int cnt=0;
    int pid=fork();
    if(pid!=0){
        printf("this is the father process!,PID=%d\n",getpid());
    }else{
       while(1){
          printf("this is the child process!,child oid=%d,father pid=%d\n",getpid(),getppid());
           sleep(2);
           if(cnt == 3){
                   exit(2);
           }
           cnt++;
        }
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值