开发中,在io密集型的场景下,我们可以使用多进程(多线程/协成更nber)来提高任务的处理速度。这就需要主进程需要等待所有工作进程执行完毕后才可以去汇总结果后退出。

但如果不规范的编写程序,就可能导致系统产生孤儿进程/僵尸进程。

父/子进程执行的流程

孤儿进程/僵尸进程

1、孤儿进程:子进程执行完毕时发现父进程已退出,子进程变成为了孤儿进程。孤儿进程后期会被系统的 init 进程接管,并 wait/waitpid 其执行状态做回收处理。对系统并无危害。

2、僵尸进程:子进程执行完毕时发现父进程未退出,会向父进程发送 SIGCHLD 信号。但父进程没有使用 wait/waitpid 或其他方式处理 SIGCHLD 信号来回收子进程,子进程变成为了对系统有害的僵尸进程。

僵尸进程无法被系统有效的回收,ps 查看时状态为 Z 的即为僵尸进程,或 top 命令可直接看到 zombie 的个数。僵尸进程的父进程此时一定仍在运行,产生僵尸进程的元凶其实是他们的父进程,杀掉父进程,僵尸进程就变为了孤儿进程,便可以转交给 init 进程回收处理。

多进程编程

在多进程开发中,我们无法统一确信父进程会先于任何子进程退出,或者不少场景父进程在创建子进程后并不会退出。这就使得我们要在父进程中处理子进程的 SIGCHLD 信号,否则就有可能产生僵尸进程。

用 PHP 来实现一个较为标准的多进程模型。

/**

* 安全的多进程处理

*/

if (!function_exists('pcntl_fork')) {

trigger_error("need pcntl extension!", E_USER_ERROR);

}

const WORKER_NUM = 4;

$workers_pid = [];

for ($i = 0; $i < WORKER_NUM; $i++) {

$pid = pcntl_fork();

if ($pid == -1) {

trigger_error("child process create failed!");

}

if ($pid == 0) {

// 子进程执行模块 编写子进程的执行逻辑

echo "I am child pid: " . getmypid() . PHP_EOL;

sleep(rand(1, 3));

exit(0);

} else {

// 父进程收集子进程的pid

$workers_pid[] = $pid;

}

}

// 父进程需使用 wait/waitpid 处理子进程的 SIGCHLD 信号 防止子进程成为僵尸进程

// 同时也可以同步父子进程的执行顺序 等待子进程全部执行完成 防止成为孤儿进程

while (true) {

// 若仍有未退出的子进程

if (!empty($workers_pid)) {

// pcntl_wait 会阻塞并等待子进程发送的信号量

$worker_pid = pcntl_wait(&$status);

if ($status == 0) {

echo 'child ' . $worker_pid . ' safe exited!' . PHP_EOL;

} else {

echo 'child ' . $worker_pid . ' wrong end with status: ' . $status . PHP_EOL;

}

// 删除子进程的 pid

$key = array_search($worker_pid, $workers_pid);

unset($workers_pid[$key]);

} else {

// 所有子进程都已执行完毕

break;

}

}

// 如果我们没有使用 wait/waitpid 处理子进程的 SIGCHLD 信号

// 且在子进程结束时父进程还未退出 则会产生僵尸进程占用系统资源

echo "main end" . PHP_EOL;

wait/waitpid 函数会阻塞父进程等待子进程发送 SIGCHLD 信号,父进程处理并回收对应的子进程,当所有的子进程回收完毕后,即 workers_pid 数组为空时,主进程正常退出即可。

形象的理解下:

父进程作为子进程的监护人,在子进程运行结束后负责清理和回收相关资源是理所当然的。

子进程在运行结束时会告诉父进程,我运行完了,把我回收掉吧,腾出地儿来。

父进程可以通过 await/awaitpid 收到子进程运行结束的信号 SIGCHLD,并回收子进程。

可有些父进程不负责任,丢下子进程直接跑掉了,子进程便成了孤儿进程,这时福利院长 init 便过来接管收留了这些子进程,让它们成为了自己的孩子,耐心的倾听它们何时执行完毕,把它们回收。

有些父进程更过分,虽然没跑路,但堵住了自己负责监听 SIGCHLD 信号的耳朵,子进程根本没办法喊应它,就变成了僵尸进程。init 看到父进程还在那,自己也不能过去接管。如果父进程一会儿突然跑掉了,那 init 可以过去接管这些子进程,因为此时这些子进程已经没有监护人,他们是孤儿进程了,init 可以接管它们。但如果父进程一直不走,那这些僵尸子进程就会一直在那里呆着占用着系统资源。

补充

1、父进程退出时子进程仍在运行,则会使得子进程变为孤儿进程,系统的 pid 为 1 的 init 进程 将会接管这些孤儿进程,待其运行结束后回收资源。

2、子进程退出时父进程仍在运行,且父进程没有对子进程发送的 SIGCHLD 信号进行处理(通常我们是调用 wait/waitpid 进行处理的),则会使得子进程成为僵尸进程 zombie,僵尸进程会继续占用系统资源。若父进程稍后退出,则僵尸进程会转为孤儿进程,进入 1 的处理流程。若父进程处于长期运行状态,则这些占用系统资源的僵尸进程会降低系统性能。

java 僵尸进程_孤儿进程与僵尸进程相关推荐

  1. vmware-vmx.exe无法结束进程_孤儿进程与僵尸进程产生原理分析,以及终极解决方案案例实现...

    开发中,在io密集型的场景下,我们可以使用多进程(多线程/协成更nber)来提高任务的处理速度.这就需要主进程需要等待所有工作进程执行完毕后才可以去汇总结果后退出. 但如果不规范的编写程序,就可能导致 ...

  2. linux查找用户前三进程_查看 Linux 系统中进程和用户的内存使用情况 | Linux 中国...

    有一些命令可以用来检查 Linux 系统中的内存使用情况,下面是一些更好的命令.-- Sandra Henry-stocker 有许多工具可以查看 Linux 系统中的内存使用情况.一些命令被广泛使用 ...

  3. kill 进程_如何查杀stopped进程

    在Linux系统下面,top命令可以查看查看stopped进程.但是不能查看stopped进程的详细信息.那么如何查看stopped 进程,并且杀掉这些stopped进程呢? ps -e j | gr ...

  4. java 关闭kafka进程_无法启动任何kafka进程

    从快速启动页面运行kafka的说明不适用于我.无法启动任何kafka进程 卡夫卡建立精细 05:55:01/kafka-0.8.1-src:58 $sbt package [info] Set cur ...

  5. 如何killtomcat进程_查看并杀掉tomcat进程

    http://blog.sina.com.cn/s/blog_5f9beca40101forf.html 打开tomcat #执行 bin/startup.sh #启动tomcat bin/shutd ...

  6. 关闭终端php就退出进程_解决windows下php-cgi进程经常自动关闭

    php-cgi在linux中有fpm管理,Apache不是采用cgi这种模式,于是乎就出现了,在windows下,采用nginx时,开启的php-cgi监听非常不稳定,经常奔溃关闭.现在解决了这个问题 ...

  7. cpu核心 线程 进程_科个普:进程、线程、并发、并行

    一.进程 刘大胖打开电脑,想写点东西,于是打开WPS,突然又想和女朋友(反正我不信)聊聊天,就又打开了微信PC端,这时操作系统就会为这两个程序生成两个进程,如图:二.线程每个进程至少包含一个线程,线程 ...

  8. 启动进程 问号_有两个这样的进程:僵尸进程amp;孤儿进程,蓝瘦香菇

    进程 先来说下什么是进程: 来看下百度是怎么说的: 光看说的不够形象,在windows系统中,它长这样: 在Mac系统中,它长这样: Linux中是这样的:(有点长截图一部分好了) [root@iz2 ...

  9. python僵尸进程和孤儿进程_进程3.0——进程状态与僵尸进程、孤儿进程

    进程3.0--进程状态与僵尸进程.孤儿进程 进程状态 一个进程的生命周期可以划分为一组状态,这些状态刻画了整个进程.进程状态即体现一个进程的生命状态 一般来说,进程有五种状态:创建状态:进程在创建时需 ...

最新文章

  1. 2021年大数据ELK(二十三):Kibana简介
  2. java构造方法可以重载吗_Java基础教程之构造器与方法重载
  3. 143. Leetcode 78. 子集 (回溯算法-子集问题)
  4. echarts tab切换_Python 数据可视化实战:使用 PyQt5 和 Echarts 打造股票数据看板
  5. 信息安全系统设计基础第八周期中学习总结—20135227黄晓妍
  6. 浅谈接口对前后端测试的意义
  7. 关于简历的一点心得(纯属个人想法,非专业人士,欢迎批评指正~)
  8. 达沃斯讨论最大的技术革命机会
  9. Ubuntu18.04之boost警告报错
  10. 实现一个简单的Tomcat 1
  11. CSS3实现的立体button
  12. android textwatcher 获取当前控件,android api解析之TextWatcher
  13. 关于adb no serial number的解决方案
  14. 数据库 使用 笛卡尔积两表联查 查询sql语句
  15. IDEA画Use Case(用例图)
  16. OCR文字识别算法总结
  17. 知乎盐选会员-share1223会员商城
  18. 由mouseover和mouseenter引发的思考
  19. 2023第十届大唐杯省赛心得体会总结
  20. 二手书交易系统用例图

热门文章

  1. mysql数据库连接jar_mysql数据库连接包
  2. 查看tomcat服务器返回的响应消息,Tomcat上的WebSocket握手返回意外响应代码500
  3. html 右下角弹窗,javascript实现的右下角弹窗实例
  4. c语言词法分析程序实验报告,实验一词法分析程序设计与实现
  5. BUUCTF(pwn)[HarekazeCTF2019]baby_rop
  6. c语言程序设计复习提纲,C语言程序设计复习提纲
  7. python方格染色_判定二部图的着色算法(Python),题,染色法,判断,二分
  8. Python 基础教程:两个list合并成一个字典的两种方法
  9. Python中的eval(),exec()以及其相关函数
  10. Python基础教程:列表字典的键值修改