调用 fork() 两次以避免僵死进程

当我们只fork()一次后,存在父进程和子进程。这时有两种方法来避免产生僵尸进程:

  • 父进程调用waitpid()等函数来接收子进程退出状态。
  • 父进程先结束,子进程则自动托管到Init进程(pid = 1)。

      目前先考虑子进程先于父进程结束的情况:     

  • 若父进程未处理子进程退出状态,在父进程退出前,子进程一直处于僵尸进程状态。
  • 若父进程调用waitpid()(这里使用阻塞调用确保子进程先于父进程结束)来等待子进程结束,将会使父进程在调用waitpid()后进入睡眠状态,只有子进程结束父进程的waitpid()才会返回。 如果存在子进程结束,但父进程还未执行到waitpid()的情况,那么这段时期子进程也将处于僵尸进程状态。

      由此,可以看出父进程与子进程有父子关系,除非保证父进程先于子进程结束或者保证父进程在子进程结束前执行waitpid(),子进程均有机会成为僵尸进程。那么如何使父进程更方便地创建不会成为僵尸进程的子进程呢?这就要用两次fork()了。

      父进程一次fork()后产生一个子进程随后立即执行waitpid(子进程pid, NULL, 0)来等待子进程结束,然后子进程fork()后产生孙子进程随后立即exit(0)。这样子进程顺利终止(父进程仅仅给子进程收尸,并不需要子进程的返回值),然后父进程继续执行。这时的孙子进程由于失去了它的父进程(即是父进程的子进程),将被转交给Init进程托管。于是父进程与孙子进程无继承关系了,它们的父进程均为Init,Init进程在其子进程结束时会自动收尸,这样也就不会产生僵尸进程了。


  1. #include <stdio.h>    
  2. #include <sys/wait.h>    
  3. #include <sys/types.h>    
  4. #include <unistd.h>    
  5.     
  6. int main(void)       
  7. {       
  8.    pid_t pid;       
  9.       
  10.     if ((pid = fork()) < 0)       
  11.     {       
  12.         fprintf(stderr,"Fork error!/n");       
  13.         exit(-1);       
  14.     }       
  15.     else if (pid == 0) /* first child */      
  16.     {        
  17.         if ((pid = fork()) < 0)       
  18.         {        
  19.             fprintf(stderr,"Fork error!/n");       
  20.             exit(-1);       
  21.         }       
  22.         else if (pid > 0)       
  23.             exit(0); /* parent from second fork == first child */      
  24.         /*    
  25.          * We're the second child; our parent becomes init as soon    
  26.          * as our real parent calls exit() in the statement above.    
  27.          * Here's where we'd continue executing, knowing that when    
  28.          * we're done, init will reap our status.    
  29.          */      
  30.         sleep(2);       
  31.         printf("Second child, parent pid = %d/n", getppid());       
  32.         exit(0);       
  33.     }       
  34.            
  35.     if (waitpid(pid, NULL, 0) != pid) /* wait for first child */      
  36.     {       
  37.         fprintf(stderr,"Waitpid error!/n");       
  38.         exit(-1);       
  39.     }       
  40.       
  41.     /*    
  42.      * We're the parent (the original process); we continue executing,    
  43.      * knowing that we're not the parent of the second child.    
  44.      */      
  45.     exit(0);       
  46. }       
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 使用系统调用fork()可以创建两个子进程。具体步骤如下: 1. 在进程调用fork()函数,该函数会返回两次,一返回子进程进程ID,另一返回。 2. 判断返回值,如果是,则说明当前进程子进程,可以在子进程中执行需要的操作;如果返回值大于,则说明当前进程进程,可以在进程中执行需要的操作。 3. 在进程中再调用fork()函数,可以创建第二个子进程,同样需要判断返回值,执行相应的操作。 4. 最终会有一个进程和两个子进程存在,可以通过进程ID来区分它们。 ### 回答2: fork()是Linux系统中的一个常见的系统调用,它用于在当前进程的基础上创建一个新的子进程,二者共享地址空间和文件描述符表,但各自有自己的栈空间和部分进程数据。 使用fork()函数可以轻松地创建两个子进程,方法是在程序中使用一个if/else语句来检测fork()的返回值,如果fork()返回0,则说明当前进程正在运行的是子进程,若fork()返回值>0,则说明当前进程正在运行的是进程。若fork()返回值<0,则表示fork()出错。 在linux系统中,每个进程都有自己的进程ID,通过fork()函数创建出来的子进程会继承当前进程的ID,但是fork()的子进程pid与进程pid不同,这是Linux系统进程间通信的一种机制。 当使用fork()函数创建了两个子进程后,我们可以通过进程创建还是子进程创建来进行进程间通信,例如进程可以向子进程发送消息,并等待子进程响应。 需要注意的是,对于子进程,它不能访问进程的数据和资源,进程也不能直接访问子进程的数据和资源,如果需要进行进程间数据传递,需要通过Linux系统提供的各种进程间通信(IPC)机制来完成。 总之,使用系统调用fork()创建两个子进程可以实现Linux系统中多进程编程的功能,为实现模块化编程、并发编程等提供了一种重要的手段。 ### 回答3: fork()是Linux系统中一个非常重要的系统调用,可以用于创建子进程。 使用fork()创建两个子进程的过程如下: 1. 在进程中使用fork()系统调用创建第一个子进程,该子进程进程的复制品,即复制进程中的代码和数据段,但是它有自己的程序计数器(PC)和栈。 2. 在第一个子进程中,我们可以使用fork()系统调用创建第二个子进程,该子进程同样是第一个子进程的复制品。 3. 进程、第一个子进程和第二个子进程拥有完全独立的地址空间,相互之间没有共享变量,只有文件描述符和信号处理程序等相同的属性。 4. 当一个进程调用fork()系统调用,操作系统会将进程的执行环境复制一个副本(叫做子进程),并让进程子进程各自执行不同的代码。 5. 在fork()系统调用中,进程会返回子进程的PID(进程ID),而子进程则返回0。 6. 当进程子进程都运行完自己的代码后,它们都可以调用exit()系统调用来正常结束自己的生命。 7. 而且,要注意fork()系统调用能够返回负值,表示发生了错误。 因此,使用系统调用fork()创建两个子进程需要注意进程复制和各进程之间的独立性,可以在不同的子进程中分别执行不同的操作,比如同处理不同的大文件。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值