PHP多进程之pcntl_fork的实例详解


Posted in PHP onOctober 15, 2017

PHP多进程编之pcntl_fork的实例详解

其实PHP是支持并发的,只是平时很少使用而已。平时使用最多的应该是使用PHP-FMP调度php进程了吧。

但是,PHP的使用并不局限于做Web,我们完全也可以使用PHP来进行系统工具类的编程,做监控或者是运维。在使用这些方向的时候,我们可以使用到PHP的更多特性,例如并发(多进程)、socket编程等。

那么接下来就说说我遇到的PHP多进程的编程。这个多进程的使用是有一个背景的,下面模糊描述一下背景。

我需要一个监控系统,当然使用PHP语言,监控系统需要监控很多种系统指标,为了让每个监控指标之间尽量专心的去做自己的事情,就需要单独使用一个进程去监控一个指标,还有一个进程去读取配置,拿到配置之后,根据配置去启动每条进程。

那么,这就需要我所说的多进程了。

  1. 首先启动一个主进程,主进程用来读取配置信息。例如,我读取到了我需要监控5个指标
  2. 接下来主进程启动5个子进程,分别监控这5个指标。
  3. 创建好5个指标监控进程之后之后,主进程进行监听配置。
  4. 一旦配置发生改变,杀死之前的进程并重新创建进程。

相对来说比较清晰的逻辑。那么接下来我们就化简一下操作:简单的说就是一个主进程创建5个子进程。

首先,创建进程在需要使用php的一个函数pcntl_fork(),这个函数可能有的同学不太熟,不过接触过Linux C变成的人都知道Linux下有个叫fork()的函数,用来创建子进程。这个函数和Linux下这个函数是一个意思。需要注意的是,这个函数在Linux下才能使用,而且需要安装pcntl的扩展。

对于这个函数怎么使用,我们可以查阅官方文档:http://php.net/manual/zh/function.pcntl-fork.php

官方文档是这样说的:

pcntl_fork()函数创建一个子进程,这个子进程仅PID(进程号) 和PPID(父进程号)与其父进程不同。fork怎样在您的系统工作的详细信息请查阅您的系统 的fork(2)手册。

成功时,在父进程执行线程内返回产生的子进程的PID,在子进程执行线程内返回0。失败时,在 父进程上下文返回-1,不会创建子进程,并且会引发一个PHP错误。

这样就可以创建一个子进程了,子进程创建成功以后会执行pcntl_fork()之后的方法。那么对于这个函数的返回值我们如何理解呢?

是这样的,我们调用函数创建进程的时候,函数执行时有时间的,而新的进程刚好是在函数执行开始和结束之间创建出来的,这样,新的进程也执行了这个函数,所以函数也需要有返回值。那么对于该函数一次执行之后,父进程和子进程都会受到该函数的返回值,由于父进程创建了子进程,而子进程并没有创建新的进程,所以子进程对于这个函数的返回结果是没有的,所以就给他赋了一个0。而父进程创建了子进程,子进程是存在pid的,所以就得到了那个进程的pid。

我们可以写个程序了解一下:

$pid = pcntl_fork();
var_dump($pid);

这个调用会输出两个值,但是我们如果直接print的只能看到一个值,也就是子进程的pid,但是使用var_dump我们就可以看到两个值,是0和子进程的pid。0这个值就是子进程返回过来的。

那么如何创建进程了解清楚之后,就可以开始创建进程了,我们需要创建5个进程,那么我就循环5次创建进程。得到如下代码:

$i=0;
 while($i!=5){
  $pid = pcntl_fork();
  echo $pid."---------hahah".$i++.PHP_EOL;
 }

这样就写好了,那么运行一下吧。啊?发现不是5个进程啊,发现有好多个进程,而且最后一个hahah4这个输出有32个,为什么是32呢?我们算一算。2^5=32,为什么最后的线程数以指数增长了呢?

想发现这个并不难,因为我们之后的每一条都执行了while循环,到最后成了进程的指数增长——也就是说fork的时候把while循环也带了进去。但是我们只是要5个进程而已。怎么办呢?

通过之前对函数的研究可以看到,子进程中会返回一个为0的值,那么我们就可以知道,0为子进程的标记。我们可以通过对子进程标记来结束进程执行。所以我们可以将我们的代码修改为如下形式:

$i=0;
while($i!=5){
 $pid = pcntl_fork();
 echo $pid."---------hahah".$i++.PHP_EOL;
 if ($pid == 0) {
  echo "子进程".PHP_EOL;
  return;
 }
}

因为0其实是对子进程的标记,那么pid这个变量在子进程里实际上是0的,所以当发现pid的值为0的时候,我们就可以断定我们当前进程为一个子进程,不需要在让他执行while并创建子进程的子进程了,所以在执行完我们的内容之后就return或者exit退出这个执行就好了。这样就能保证我们执行创建了5个进程而不是32个了。

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

PHP 相关文章推荐
discuz论坛 用户登录 后台程序代码
Nov 27 PHP
一个PHP分页类的代码
May 18 PHP
工厂模式在Zend Framework中应用介绍
Jul 10 PHP
支持中文和其他编码的php截取字符串函数分享(截取中文字符串)
Mar 13 PHP
适用于抽奖程序、随机广告的PHP概率算法实例
Apr 09 PHP
php实现的漂亮分页方法
Apr 17 PHP
PHP操作文件的一些基本函数使用示例
Nov 18 PHP
php通过rmdir删除目录的简单用法
Mar 18 PHP
PHP实现163邮箱自动发送邮件
Mar 29 PHP
PHP载入图像imagecreatefrom_gif_jpeg_png系列函数用法分析
Nov 14 PHP
PHP实现QQ登录的开原理和实现过程
Feb 04 PHP
详解Laravel5.6 Passport实现Api接口认证
Jul 27 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
laravel 5异常错误:FatalErrorException in Handler.php line 38的解决
Oct 12 #PHP
You might like
在PHP中使用模板的方法
2008/05/24 PHP
thinkPHP事务操作简单案例分析
2019/10/17 PHP
PHP使用Http Post请求发送Json对象数据代码解析
2020/07/16 PHP
jquery 获取json数据实现代码
2009/04/27 Javascript
firefox事件处理之自动查找event的函数(用于onclick=foo())
2010/08/05 Javascript
jquery获取下拉列表的值为null的解决方法
2011/03/18 Javascript
MultiSelect左右选择控件的设计与实现介绍
2013/06/08 Javascript
使用jQuery不判断浏览器高度解决iframe自适应高度问题
2014/12/16 Javascript
深入剖析JavaScript中的函数currying柯里化
2016/04/29 Javascript
JS制作图形验证码实现代码
2020/10/19 Javascript
一个例子轻松学会Vue.js
2017/01/02 Javascript
js 监控iframe URL的变化实例代码
2017/07/12 Javascript
Postman模拟发送带token的请求方法
2018/03/31 Javascript
vue 1.0 结合animate.css定义动画效果
2018/07/11 Javascript
Vue将页面导出为图片或者PDF
2020/08/17 Javascript
在JavaScript中如何访问暂未存在的嵌套对象
2019/06/18 Javascript
Vue实现滑动拼图验证码功能
2019/09/15 Javascript
浅谈vue-router路由切换 组件重用挖下的坑
2019/11/01 Javascript
Windows上node.js的多版本管理工具用法实例分析
2019/11/06 Javascript
vue父子组件的通信方法(实例详解)
2019/11/10 Javascript
jquery css实现流程进度条
2020/03/26 jQuery
浅谈python中的__init__、__new__和__call__方法
2017/07/18 Python
详解Python爬取并下载《电影天堂》3千多部电影
2019/04/26 Python
python实现websocket的客户端压力测试
2019/06/25 Python
python输出电脑上所有的串口名的方法
2019/07/02 Python
Python中拆分字符串的操作方法
2019/07/23 Python
python bluetooth蓝牙信息获取蓝牙设备类型的方法
2019/11/29 Python
pytorch中nn.Conv1d的用法详解
2019/12/31 Python
canvas进阶之贝塞尔公式推导与物体跟随复杂曲线的轨迹运动
2018/01/10 HTML / CSS
Philosophy美国官网:美国美容品牌
2016/08/15 全球购物
汽车运用工程毕业生自荐信
2013/10/29 职场文书
会计电算化学生个人的自我评价
2014/02/08 职场文书
校园主题婚礼活动策划方案
2014/09/15 职场文书
流动人口婚育证明范本
2014/09/26 职场文书
师德师风心得体会(2016精选篇)
2016/01/12 职场文书
Ajax实现异步加载数据
2021/11/17 Javascript