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 相关文章推荐
防止MySQL注入或HTML表单滥用的PHP程序
Jan 21 PHP
php 遍历数据表数据并列表横向排列的代码
Sep 05 PHP
国外比较好的几个的Php开源建站平台小结
Apr 22 PHP
php简单的会话类代码
Aug 08 PHP
PHP变量的定义、可变变量、变量引用、销毁方法
Dec 20 PHP
PHP中substr()与explode()函数用法分析
Nov 24 PHP
PHP日期函数date格式化UNIX时间的方法
Mar 19 PHP
一个完整的PHP类包含的七种语法说明
Jun 04 PHP
php示例详解Constructor Prototype Pattern 原型模式
Oct 15 PHP
php外部执行命令函数用法小结
Oct 11 PHP
php获取小程序码的实现代码(B类接口)
Jun 13 PHP
如何在PHP环境中使用ProtoBuf数据格式
Jun 19 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 无限级缓存的类的扩展
2009/03/16 PHP
需要发散思维学习PHP
2009/06/29 PHP
php HandlerSocket的使用
2011/05/02 PHP
php的4种常见运行方式
2015/03/20 PHP
使用ltrace工具跟踪PHP库函数调用的方法
2016/04/25 PHP
深入理解JavaScript系列(15) 函数(Functions)
2012/04/12 Javascript
变量声明时命名与变量作为对象属性时命名的区别解析
2013/12/06 Javascript
jQuery绑定事件不执行但alert后可以正常执行
2014/06/03 Javascript
Web前端开发工具——bower依赖包管理工具
2016/03/29 Javascript
浅谈Javascript中的函数、this以及原型
2016/10/09 Javascript
AngularJS使用ng-repeat和ng-if实现数据的删选显示效果示例【适用于表单数据的显示】
2016/12/13 Javascript
javascript滚轮事件基础实例讲解(37)
2017/02/14 Javascript
angular2中router路由跳转navigate的使用与刷新页面问题详解
2017/05/07 Javascript
vue加载完成后的回调函数方法
2018/09/07 Javascript
使用vue 国际化i18n 实现多实现语言切换功能
2018/10/11 Javascript
jQuery实现的记住帐号密码功能完整示例
2019/08/03 jQuery
VSCode launch.json配置详细教程
2020/06/18 Javascript
jQuery实现图片切换效果
2020/10/19 jQuery
Python类的基础入门知识
2008/11/24 Python
Python 3.x 连接数据库示例(pymysql 方式)
2017/01/19 Python
django的ORM操作 增加和查询
2019/07/26 Python
pytorch程序异常后删除占用的显存操作
2020/01/13 Python
python操作yaml说明
2020/04/08 Python
python+adb命令实现自动刷视频脚本案例
2020/04/23 Python
Python实现爬取并分析电商评论
2020/06/19 Python
python 实现弹球游戏的示例代码
2020/11/17 Python
python中使用np.delete()的实例方法
2021/02/01 Python
美国高端牛仔品牌:Silver Jeans
2019/12/12 全球购物
学校七一活动方案
2014/01/19 职场文书
二年级语文教学反思
2014/02/02 职场文书
个人社会实践自我鉴定
2014/03/24 职场文书
厨师个人自我鉴定范文
2014/04/19 职场文书
小学生五年级大队长竞选发言稿
2014/09/12 职场文书
《火烧云》教学反思
2016/02/23 职场文书
2019中小学生安全过暑期倡议书
2019/06/24 职场文书
nginx如何将http访问的网站改成https访问
2021/03/31 Servers