PHP多进程编程之僵尸进程问题的理解


Posted in PHP onOctober 15, 2017

PHP多进程编程之僵尸进程问题的理解

使用pcntl_fork函数可以让PHP实现多进程并发或者异步处理的效果:https://3water.com/article/125789.htm

那么问题是我们产生的进程需要去控制,而不能置之不理。最基本的方式就是fork进程和杀死进程。

通过利用pcntl_fork函数,我们已经有了新的子进程,而子进程接下来完成我们需要处理的内容,那么我们就暂且叫做service()吧,而且我们需要很多个service()进行处理,再次参照我们之前的需求,父进程需要一直循环读取配置文件,等待文件发生改变。通过对pcntl_fork的方式,很容易我们就可以写出如下代码:

$res = config();
//kill进程
for($i = 0; $i < $res[sum]; $i++) {
  $pid = pcntl_fork();
  if ($pid == 0) {
    service();
    return;
  }
}

代码中注释的地方我们需要在配置文件中发生改变的时候杀死进程,杀死进程的的方式很简单,可以使用kill命令直接杀死,比如(假设pid为123):

1 kill 123

但是我们发现,使用这个杀死进程的方式并没有真正的把进程杀死,这个子进程被杀死后还占用这个进程的资源,我们成为僵尸进程,僵尸进程是使用kill命令无法杀死的。想要解决这个问题,我们能做的只有两种方式。

1. shutdown

2. 杀死该进程的父进程。

但是这两种方法都不行,因为这个程序的目的是监控常驻在服务器内,服务器不能关闭,并且父进程也不能被干掉。这时候我们看到了官方文档对于fork方法的解释:

pcntl_wait($status); //等待子进程中断,防止子进程成为僵尸进程。

原来有种方式可以防止进程成为僵尸进程,但是,官网给出的代码是这样子的:

$pid = pcntl_fork();
//父进程和子进程都会执行下面代码
if ($pid == -1) {
  //错误处理:创建子进程失败时返回-1.
   die('could not fork');
} else if ($pid) {
   //父进程会得到子进程号,所以这里是父进程执行的逻辑
   pcntl_wait($status); //等待子进程中断,防止子进程成为僵尸进程。
} else {
   //子进程得到的$pid为0, 所以这里是子进程执行的逻辑。
}

什么意思呢?就是父进程会等待子进程运行,等子进程运行结束之后,才会进行下一步,并且也会消除僵尸进程。但是这里又和我们的需求不符合了,我们的子进程为一个死循环的程序,不断的查找输出,更本没有结束的时候,并且我们需要的是异步处理而不是同步。但是这个方法可以用吗?其实当然可以。

在pcntl_wait的文档中是这么解释这个函数的:

wait函数刮起当前进程的执行直到一个子进程退出或接收到一个信号要求中断当前进程或调用一个信号处理函数。 如果一个子进程在调用此函数时已经退出(俗称僵尸进程),此函数立刻返回。子进程使用的所有系统资源将 被释放。关于wait在您系统上工作的详细规范请查看您系统的wait(2)手册。

我们发现,当这个函数发现子进程成为了僵尸进程就会释放僵尸进程的资源——前提是这个僵尸进程为这个父进程的子进程。那么我们就可以巧妙的利用这个方式让这些僵尸进程释放资源了,所以就有了如下代码:

posix_kill(123, 9);
 pcntl_wait($status);

这样我们先使用kill干掉这个进程,这个进程就不会再运行了,但是这个进程成为了僵尸进程,占用着资源,我们下一句就执行一次pcntl_wait()让这些僵尸进程释放资源,这样,子进程才真正的被终止了,僵尸进程被消除了。

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

PHP 相关文章推荐
cmd下运行php脚本
Nov 25 PHP
PHP 读取文件的正确方法
Apr 29 PHP
php 应用程序安全防范技术研究
Sep 25 PHP
那些年我们错过的魔术方法(Magic Methods)
Jan 14 PHP
PHP遍历文件夹与文件类及处理类用法实例
Sep 23 PHP
php实现兼容2038年后Unix时间戳转换函数
Mar 18 PHP
PHP安全上传图片的方法
Mar 21 PHP
PHP中使用socket方式GET、POST数据实例
Apr 02 PHP
简单解析PHP程序的运行流程
Jun 23 PHP
PHP图片水印类的封装
Jul 06 PHP
PHP实现的服务器一致性hash分布算法示例
Aug 09 PHP
Laravel Validator自定义错误返回提示消息并在前端展示
May 09 PHP
PHP多进程之pcntl_fork的实例详解
Oct 15 #PHP
详解thinkphp5+swoole实现异步邮件群发(SMTP方式)
Oct 13 #PHP
详解PHP字符串替换str_replace()函数四种用法
Oct 13 #PHP
浅谈PHP接入(第三方登录)QQ登录 OAuth2.0 过程中遇到的坑
Oct 13 #PHP
PHP长网址与短网址的实现方法
Oct 13 #PHP
如何直接访问php实例对象中的private属性详解
Oct 12 #PHP
thinkphp中的多表关联查询的实例详解
Oct 12 #PHP
You might like
杏林同学录(六)
2006/10/09 PHP
PHP 工厂模式使用方法
2010/05/18 PHP
php 实现进制转换(二进制、八进制、十六进制)互相转换实现代码
2010/10/22 PHP
PHP自动识别字符集并完成转码详解
2013/08/02 PHP
PHP实现的通过参数生成MYSQL语句类完整实例
2016/04/11 PHP
通过chrome浏览器控制台(Console)进行PHP Debug的方法
2016/10/19 PHP
使用composer命令加载vendor中的第三方类库 的方法
2019/07/09 PHP
JS 面向对象的5钟写法
2009/07/31 Javascript
jQuery实现 注册时选择阅读条款 左右移动
2013/04/11 Javascript
jquery垂直公告滚动实现代码
2013/12/08 Javascript
JavaScript实现Iterator模式实例分析
2015/06/09 Javascript
使用js复制链接中的部分文字的方法
2015/07/30 Javascript
js实现搜索框关键字智能匹配代码
2020/03/26 Javascript
JavaScript的React Web库的理念剖析及基础上手指南
2016/05/10 Javascript
整理一些最近经常遇到的前端面试题
2017/04/25 Javascript
JS ES6多行字符串与连接字符串的表示方法
2017/04/26 Javascript
JavaScript数据结构学习之数组、栈与队列
2017/05/02 Javascript
AngularJS实现的JSONP跨域访问数据传输功能详解
2017/07/20 Javascript
浅谈如何使用webpack构建多页面应用
2018/05/30 Javascript
vue移动端下拉刷新和上拉加载的实现代码
2018/09/08 Javascript
vue项目中使用vue-i18n报错的解决方法
2019/01/13 Javascript
bootstrap中的导航条实例代码详解
2019/05/20 Javascript
Webpack中loader打包各种文件的方法实例
2019/09/03 Javascript
IE11下CKEditor在Bootstrap Modal中下拉问题的解决
2019/09/25 Javascript
mapboxgl实现带箭头轨迹线的代码
2021/01/04 Javascript
[50:28]2018DOTA2亚洲邀请赛 3.31 小组赛 A组 Newbee vs KG
2018/04/01 DOTA
用Python的Flask框架结合MySQL写一个内存监控程序
2015/11/07 Python
python通过tcp发送xml报文的方法
2018/12/28 Python
在python 中实现运行多条shell命令
2019/01/07 Python
keras实现多GPU或指定GPU的使用介绍
2020/06/17 Python
h5页面背景图很长要有滚动条滑动效果的实现
2021/01/27 HTML / CSS
Guess荷兰官网:美国服饰品牌
2020/01/22 全球购物
电子商务网站的创业计划书
2014/01/05 职场文书
关于开学的感想
2015/08/10 职场文书
避坑之 JavaScript 中的toFixed()和正则表达式
2022/04/19 Javascript
html5+实现plus.io进行拍照和图片等获取
2022/06/01 HTML / CSS