父进程结束后,子进程自动结束

5三/12 27

Linux下让父进程结束后,子进程自动结束


在多进程编程的时候,经常会遇到这样的情况。父进程创建了一堆子进程,当遇到错误或者操作失误的时候把父进程关闭了,但是子进程还在跑,不得不一个一个地杀死子进程,或者使用ps,grep,awk,kill来配合批量杀死。

之前在写 xxfpm(一个PHP-CGI的进程管理) 的时候,在Linux下使用父进程给子进程信号通知的方式来达到用户杀死父进程时,子进程也随即关闭。但是这种方法不太完美。例如,如果父进程被KILL信号杀死,完全没有机会给子进程发送信号了。

在网上搜了一下,用Linux下libc的prctl设置PR_SET_PDEATHSIG属性,似乎可以让子进程在父进程自动结束后接收到信号。这个方法似乎很完美!!!

PR_SET_PDEATHSIG (since Linux 2.1.57)
              Set the parent process death signal of the calling process to arg2
              (either a signal value in the range 1..maxsig, or 0 to clear).  This is
              the signal that the calling process will get when its parent dies.
              This value is cleared for the child of a fork(2).

测试代码:

  1. #!/usr/bin/env python  
  2.   
  3. import os  
  4. import ctypes  
  5. import time  
  6.   
  7. libc = ctypes.CDLL('libc.so.6')  
  8.   
  9. for i in xrange(4):  
  10.     pid = os.fork()  
  11.     if pid == 0:  
  12.         libc.prctl(115)  
  13.         while True:  
  14.             print 'Child:', i  
  15.             time.sleep(1)  
  16.         raise SystemExit  
  17.   
  18. print 'Wait for 10 sec...'  
  19. time.sleep(10)  
  20. print 'Exit'  
转载完毕,留下点问题:刚接触python 大概能看懂,但是c语言或者其他的怎么弄呢?
<think>嗯,用户问的是POSIX中是否有办法在父进程结束自动杀死所有子进程。我需要先回忆一下POSIX的相关标准以及进程管理的机制。 首先,POSIX本身有没有直接提供这样的机制呢?我记得当父进程终止时,子进程并不会被系统自动终止,而是会成为孤儿进程,被init进程(PID 1)接管。这时候子进程会继续运行,除非它们自己退出或者被显式终止。所以默认情况下,父进程结束不会杀死子进程,这一点需要明确告诉用户。 那用户的需求可能是在父进程意外终止或者正常退出时,确保所有子进程也被终止。这时候需要想办法实现这一点。常见的解决方法可能有几种: 一种是父进程在创建子进程后,注册一个信号处理函数,比如处理SIGTERM或SIGINT,当父进程收到这些信号时,主动向所有子进程发送终止信号。但这种方法的问题是无法处理父进程崩溃或者被SIGKILL杀死的情况,因为SIGKILL无法被捕获,这时候父进程无法执行任何清理操作。 另一个方法是使用进程组(process group)。父进程可以创建一个新的进程组,并将所有子进程放入该组中。然后,当父进程需要终止时,可以向整个进程组发送信号。但同样,如果父进程是被强制终止的,比如SIGKILL,这时候如何触发发送信号呢?这时候可能需要依赖系统的机制,比如使用控制终端(controlling terminal)的SIGHUP信号。当终端关闭时,系统会发送SIGHUP给进程组,但这需要进程在同一个会话中,并且依赖于终端的设置。 还有一种方法是使用prctl()系统调用,设置子进程父进程退出时收到SIGKILL。不过这个函数是Linux特有的,不是POSIX标准的一部分,所以可能不符合用户的需求,因为用户明确问的是POSIX中的方法。 那回到POSIX标准,是否有推荐的做法?可能需要结合进程组和信号处理。父进程可以设置自己的进程组,并在终止时发送信号给整个组。但如何确保父进程终止时能触发这个操作呢?如果父进程正常退出,可以在退出前发送信号。但如果父进程异常终止,可能需要依赖系统机制。 另外,当父进程终止时,子进程会成为孤儿进程,由init接管。这时候,如果子进程继续运行,可能需要其他方式终止它们。例如,某些系统在用户注销时会发送SIGHUP给进程,但这也依赖于具体实现,不是POSIX标准。 总结起来,POSIX本身没有直接的方法在父进程结束自动杀死所有子进程。但可以通过以下方法间接实现: 1. 父进程正常退出时,主动发送终止信号给子进程。 2. 使用进程组,当父进程终止时,由其他进程(如shell)发送信号给整个进程组。 3. 利用终端会话的SIGHUP信号,如果进程在同一个会话中,当终端关闭时,子进程会收到信号。 用户可能需要根据具体情况选择合适的方法,同时要注意这些方法的局限性和平台依赖性。例如,使用进程组和shell命令结合,如kill -- -PGID,但需要确保父进程能够正确设置进程组,并在终止时触发相应的操作。 另外,需要提醒用户注意,如果父进程被SIGKILL杀死,则无法执行任何清理操作,这时候只能依赖系统机制或外部监控进程来处理孤儿进程。</think>在POSIX标准中,没有直接提供父进程退出时自动杀死所有子进程的内置机制。但可通过以下方法间接实现: --- ### 一、默认行为与局限性 1. **孤儿进程** 父进程终止后,子进程会成为**孤儿进程**,由`init`进程(PID=1)接管。 这些子进程会继续运行,除非它们自行退出或被显式终止。 2. **关键限制** 若父进程被`SIGKILL`强制终止或意外崩溃,无法主动清理子进程。 --- ### 二、间接实现方案 #### 方案1:父进程主动终止子进程(正常退出场景) ```c #include <signal.h> #include <sys/wait.h> void cleanup_children() { // 向所有子进程发送SIGTERM kill(0, SIGTERM); // 等待子进程退出 while (wait(NULL) > 0); } int main() { // 注册退出清理函数 atexit(cleanup_children); // 创建子进程... fork(); // 父进程正常退出时,cleanup_children()会被调用 return 0; } ``` **局限性**:无法处理`SIGKILL`或崩溃等异常退出场景。 --- #### 方案2:进程组 + Shell辅助 1. **父进程创建新进程组** ```c setpgid(0, 0); // 父进程自建进程组 fork(); // 子进程默认继承进程组 ``` 2. **通过Shell终止整个进程组** ```bash # 启动父进程时绑定进程组 ./parent_process & # 父进程退出后,手动终止整个进程组 kill -- -$(pgrep -o parent_process) ``` **优势**:可彻底终止同一进程组内的所有进程。 --- #### 方案3:终端会话关联(依赖环境) 若进程在终端中运行,终端关闭时会发送`SIGHUP`信号给关联进程组。 可通过`nohup`或`disown`解除关联,但需谨慎使用。 --- ### 三、非POSIX扩展方案 - **Linux专属**: `prctl(PR_SET_PDEATHSIG, SIGKILL)` 子进程调用此函数后,会在父进程退出时收到`SIGKILL`。 ```c #include <sys/prctl.h> prctl(PR_SET_PDEATHSIG, SIGKILL); ``` --- ### 四、总结 | 场景 | 推荐方案 | |--------------------|---------------------------------| | 父进程正常退出 | 方案1(主动清理) | | 需强制终止所有子进程 | 方案2(进程组 + Shell辅助) | | Linux环境 | `prctl(PR_SET_PDEATHSIG)`扩展 | **注意**:若需严格遵循POSIX,建议结合进程组管理与Shell脚本控制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值