对于用PHP进行多进程并发编程,不可避免要遇到僵尸进程的问题。

僵尸进程是指的父进程已经退出,而该进程dead之后没有进程接受,就成为僵尸进程(zombie)进程。任何进程在退出前(使用exit退出) 都会变成僵尸进程(用于保存进程的状态等信息),然后由init进程接管。如果不及时回收僵尸进程,那么它在系统中就会占用一个进程表项,如果这种僵尸进程过多,最后系统就没有可以用的进程表项,于是也无法再运行其它的程序。
 
方法一:
父进程通过pcntl_wait和pcntl_waitpid等函数等待子进程结束
  1. $pid = pcntl_fork();
  2.  
  3. if($pid == -1) {
  4. die('fork error');
  5. } else if ($pid) {
  6. //父进程阻塞着等待子进程的退出
  7. //pcntl_wait($status);
  8.  
  9. //pcntl_waitpid($pid, $status);
  10.  
  11. //非阻塞方式
  12. //pcntl_wait($status, WNOHANG);
  13.  
  14. //pcntl_waitpid($pid, $status, WNOHANG);
  15. } else {
  16. sleep(3);
  17. echo "child \r\n";
  18. exit;
  19. }
方法二:
可以用signal函数为SIGCHLD安装handler,因为子进程结束后,父进程会收到该信号,可以在handler中调用pcntl_wait或pcntl_waitpid来回收。
  1. <?php
  2. declare(ticks = 1);
  3.  
  4. //信号处理函数
  5. function sig_func() {
  6. echo "SIGCHLD \r\n";
  7. pcntl_wait($status);
  8.  
  9. //pcntl_waitpid(-1, $status);
  10.  
  11. //非阻塞
  12. //pcntl_wait($status, WNOHANG);
  13. //pcntl_waitpid(-1, $status, WNOHANG);
  14. }
  15.  
  16. pcntl_signal(SIGCHLD, 'sig_func');
  17.  
  18. $pid = pcntl_fork();
  19.  
  20. if($pid == -1) {
  21. die('fork error');
  22. } else if ($pid) {
  23. sleep(10);
  24. } else {
  25. sleep(3);
  26. echo "child \r\n";
  27. exit;
  28. }
如果子进程还没有结束时,父进程就结束了,那么init进程会自动接手这个子进程,进行回收。
如果父进程是循环,又没有安装SIGCHLD信号处理函数调用wait或waitpid()等待子进程结束。那么子进程结束后,没有回收,就产生僵尸进程了。
例如:
  1. <?php
  2. $pid = pcntl_fork();
  3.  
  4. if($pid == -1) {
  5. die('fork error');
  6. } else if ($pid) {
  7. for(;;) {
  8. sleep(3);
  9. }
  10. } else {
  11. echo "child \r\n";
  12. exit;
  13. }

父进程是个死循环,也没有安装SIGCHLD信号处理函数,子进程结束后。我们通过如下命令查看  > ps -A -o stat,ppid,pid,cmd | grep -e '^[Zz]'

会发现一个僵尸进程。
 
代码改进一下:
  1. <?php
  2. declare(ticks = 1);
  3.  
  4. //信号处理函数
  5. function sig_func() {
  6. echo "SIGCHLD \r\n";
  7.  
  8. pcntl_waitpid(-1, $status, WNOHANG);
  9. }
  10.  
  11. pcntl_signal(SIGCHLD, 'sig_func');
  12.  
  13. $pid = pcntl_fork();
  14.  
  15. if($pid == -1) {
  16. die('fork error');
  17. } else if ($pid) {
  18. for(;;) {
  19. sleep(3);
  20. }
  21. } else {
  22. echo "child \r\n";
  23. exit;
  24. }
当子进程结束后,再通过命令查看时,我们发现这时就没有僵尸进程了,这说明父进程对它进行了回收。
 
方法三:
如果父进程不关心子进程什么时候结束,那么可以用pcntl_signal(SIGCHLD, SIG_IGN)通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不再给父进程发送信号。
  1. <?php
  2. declare(ticks = 1);
  3.  
  4. pcntl_signal(SIGCHLD, SIG_IGN);
  5.  
  6. $pid = pcntl_fork();
  7.  
  8. if($pid == -1) {
  9. die('fork error');
  10. } else if ($pid) {
  11. for(;;) {
  12. sleep(3);
  13. }
  14. } else {
  15. echo "child \r\n";
  16. exit;
  17. }
当子进程结束后,SIGCHLD信号并不会发送给父进程,而是通知内核对子进程进行了回收。
 
方法四:
通过pcntl_fork两次,也就是父进程fork出子进程,然后子进程中再fork出孙进程,这时子进程退出。那么init进程会接管孙进程,孙进程退出后,init会回收。不过子进程还是需要父进程进行回收。我们把业务逻辑放到孙进程中执行,父进程就不需要pcntl_wait或pcntl_waitpid来等待孙进程(即业务进程)。
  1. <?php
  2. $pid = pcntl_fork();
  3.  
  4. if($pid == -1) {
  5. die('fork error');
  6. } else if ($pid) {
  7. //父进程等待子进程退出
  8. pcntl_wait($status);
  9. echo "parent \r\n";
  10. } else {
  11. //子进程再fork一次,产生孙进程
  12. $cpid = pcntl_fork();
  13. if($cpid == -1) {
  14. die('fork error');
  15. } else if ($cpid) {
  16. //这里是子进程,直接退出
  17. echo "child \r\n";
  18. exit;
  19. } else {
  20. //这里是孙进程,处理业务逻辑
  21. for($i = 0; $i < 10; ++$i) {
  22. echo "work... \r\n";
  23. sleep(3);
  24. }
  25. }
  26. }

子进程退出后,父进程回收子进程,孙进程继续业务逻辑的处理。当孙进程也执行完毕退出后,init回收孙进程。

php多进程防止出现僵尸进程的更多相关文章

  1. PHP多进程编之僵尸进程问题

    上一篇说到了使用pcntl_fork函数可以让PHP实现多进程并发或者异步处理的效果.那么问题是我们产生的进程需要去控制,而不能置之不理.最基本的方式就是fork进程和杀死进程. 通过利用pcntl_ ...

  2. php多进程pcntl学习-僵尸进程

    上个月写的文章,php多进程pcntl学习(一)现在发现并不完整,因为虽然提到了关闭子进程,但是并没有回收子进程,简单的说就是当子进程比父进程先退出,而父进程没对其做任何处理的时候,子进程将会变成僵尸 ...

  3. PHP多进程编程之僵尸进程问题

    上一篇说到了使用pcntl_fork函数可以让PHP实现多进程并发或者异步处理的效果.那么问题是我们产生的进程需要去控制,而不能置之不理.最基本的方式就是fork进程和杀死进程. 通过利用pcntl_ ...

  4. php多进程 防止出现僵尸进程

    对于用PHP进行多进程并发编程,不可避免要遇到僵尸进程的问题. 僵尸进程是指的父进程已经退出,而该进程dead之后没有进程接受,就成为僵尸进程(zombie)进程.任何进程在退出前(使用exit退出) ...

  5. 多进程wait、僵尸进程、孤儿进程、prctl

    1.概念 1.孤儿进程:一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程.孤儿进程将被init进程(进程号为1)所收养,从而保证每个进程都会有一个父进程.而Init进程会自 ...

  6. PHP的多进程--防止僵尸进程(转)

    原文地址:http://twei.site/2017/08/08/PHP%E7%9A%84%E5%A4%9A%E8%BF%9B%E7%A8%8B-%E9%98%B2%E6%AD%A2%E5%83%B5 ...

  7. Linux 网络编程详解六(多进程服务器僵尸进程解决方案)

    小结:在点对点p2p程序中,服务器端子程序退出,子进程会主动发送信号,关闭父进程,但是这种模式导致服务器只能支持一个客户端连接,本章节中使用新的框架,子进程退出,不主动发送信号关闭父进程,而是父进程安 ...

  8. python学习笔记—— 多进程中的 孤儿进程和僵尸进程

    1 基本概述 1.1 孤儿进程和僵尸进程 父进程创建子进程后,较为理想状态是子进程结束,父进程回收子进程并释放子进程占有的资源:而实际上,父子进程是异步过程,两者谁先结束是无顺的,一般可以通过父进程调 ...

  9. C/C++网络编程8——多进程服务器端之销毁僵尸进程

    上一节提到,当子进程执行结束,父进程还在执行,在父进程结束之前子进程会成为僵尸进程,那么怎么销毁僵尸进程呢?父进程主动接收子进程的返回值. 销毁僵尸进程的方法: 1:使用wait函数 2:使用wait ...

随机推荐

  1. 关于linux中使用vim打开文件出现^M的解决方法

    在linux下,不可避免的会用VIM打开一些windows下编辑过的文本文件.我们会发现文件的每行结尾都会有一个^M符号,这是因为 DOS下的编辑器和Linux编辑器对文件行末的回车符处理不一致, 各 ...

  2. ArcGIS api for javascript——加载查询结果,悬停显示信息窗口

    转自原文 ArcGIS api for javascript——加载查询结果,悬停显示信息窗口 描述 本例在开始和地图交互前执行一个查询任务并加在查询结果.这允许用户鼠标悬停在任意郡县时立即见到Inf ...

  3. jQuery Validation让验证变得如此easy(三)

    下面代码进行对jQuery Validation的简单演示包含必填项.字符长度,格式验证 一.引入文件 <script src="js/jquery-1.8.0.min.js" ...

  4. 火云开发课堂 - 《使用Cocos2d-x 开发3D游戏》系列 第七节:PS基础:UV动画

    <使用Cocos2d-x 开发3D游戏>系列在线课程 第七节:PS基础:UV动画 视频地址:http://edu.csdn.net/course/attend/1330/20807 交流论 ...

  5. 熊猫猪新系统測试之中的一个:Windows 10 技术预览版

    话说本猫不用windows非常多年了呀! 只是看到微软最新的Windows10还是手痒了.想安装体验一把. 于是第一时间下载,并做成usb引导安装镜像,在08年的老台式机上安装尝鲜鸟.下载ISO和安装 ...

  6. HDU 3656 二分+dlx判定

    Fire station Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) To ...

  7. linux下获取按键响应事件【转】

    本文转载自:https://my.oschina.net/u/157503/blog/91548 1.问题 通过一个死循环将读取键盘对应的设备文件将触发键盘事件在屏幕上打印出来,按esc退出程序 代码 ...

  8. Coursera Algorithms week2 基础排序 练习测验: Permutation

    题目原文: Given two integer arrays of size n , design a subquadratic algorithm to determine whether one ...

  9. Wall(凸包)

    http://poj.org/problem?id=1113 题意:给出一些点的坐标,和一个半径r,求出这些点围成的凸包的周长再加上一个半径为r的圆的周长. #include <stdio.h& ...

  10. js基本功能大全

    1.javascript的数组API: //定义数组 var pageIds = new Array(); pageIds.push('A'); 数组长度 pageIds.length; //shif ...