PHP多进程系列笔(转)
本系列文章将向大家讲解pcntl_*
系列函数,从而更深入的理解进程相关知识。
PCNTL在PHP中进程控制支持默认是关闭的。您需要使用
--enable-pcntl
配置选项重新编译PHP的 CGI或CLI版本以打开进程控制支持。
如果自带的PHP没有安装pcntl扩展,可以下载相同版本的源码,进入ext/pcntl
使用phpize
编译安装。
Note: 此扩展在 Windows 平台上不可用。
pcntl_fork
int pcntl_fork ( void )
用于创建子进程。成功时,在父进程执行线程内返回产生的子进程的PID,在子进程执行线程内返回0。失败时,在父进程上下文返回-1,不会创建子进程,并且会引发一个PHP错误。
fork.php
<?php
$pid = pcntl_fork();
if($pid == -1){
//错误处理:创建子进程失败时返回-1.
die( 'could not fork' );
}elseif($pid){
//父进程会得到子进程号,所以这里是父进程执行的逻辑
$id = getmypid();
echo "Parent process,pid {$id}, child pid {$pid}\n";
}else{
//子进程得到的$pid为0, 所以这里是子进程执行的逻辑
$id = getmypid();
echo "Child process,pid {$id}\n";
sleep(10);
}
命令行运行:
$ php fork.php
Parent process,pid 98, child pid 99
Child process,pid 99
该例里父进程还没有来得及等子进程运行完毕就自动退出了,子进程由 init
进程接管。通过 ps -ef | grep php
看到子进程还在运行:
[root@9355490fe5da /]# ps -ef | grep php
root 105 1 0 16:46 pts/0 00:00:00 php fork.php
root 107 27 0 16:46 pts/1 00:00:00 grep php
子进程成为孤立进程,ppid(父进程id)变成1了。如果在父进程里也加个sleep(5)
,你会看到子进程ppid本来是大于1的,后来就变成1了。
注:如果是docker环境,孤立进程的ppid可能是0。
pcntl_wait
pcntl_wait()
函数用来让父进程等待子进程退出,默认情况下会阻塞主进程。
阻塞模式
紧接着上面的例子,如果想等子进程运行结束后父进程再退出,该怎么办?那就用到pcntl_wait
了。
int pcntl_wait ( int &$status [, int $options = 0 ] )
该函数阻塞当前进程,只到当前进程的一个子进程退出或者收到一个结束当前进程的信号。
我们修改代码:
<?php
$pid = pcntl_fork();
if($pid == -1){
exit("fork fail");
}elseif($pid){
$id = getmypid();
echo "Parent process,pid {$id}, child pid {$pid}\n";
pcntl_wait($status);
//pcntl_waitpid($pid, $status);
}else{
$id = getmypid();
echo "Child process,pid {$id}\n";
sleep(10);
}
此时再次运行程序,父进程就会一直等待子进程运行结束然后退出。
pcntl_waitpid()
和pcntl_wait()
功能相同。前者第一个参数支持指定pid参数,当指定-1作为pid
的值等同于后者。
int pcntl_waitpid ( int $pid , int &$status [, int $options = 0 ] )
当已知子进程pid的时候,可以使用
pcntl_waitpid()
。
这两个函数返回退出的子进程进程号(>1),发生错误时返回-1,如果提供了 WNOHANG
作为option(wait3可用的系统)并且没有可用子进程时返回0。
返回值为退出的子进程进程号时,想了解如何退出,可以通过 $status
状态码反应。
非阻塞模式
pcntl_wait()
默认情况下会阻塞主进程,直到子进程执行完毕才继续往下运行。如果设置最后一个参数为常量WNOHANG
,那么就不会阻塞主进程,而是继续执行后续代码, 此时 pcntl_waitpid
就会返回0。
示例:
<?php
$pid = pcntl_fork();
if($pid == -1){
exit("fork fail");
}elseif($pid){
$id = getmypid();
echo "Parent process,pid {$id}, child pid {$pid}\n";
while(1){
$res = pcntl_wait($status, WNOHANG);
//$res = pcntl_waitpid($pid, $status, WNOHANG);
if ($res == -1 || $res > 0){
sleep(10);//此处为了方便看效果,实际不需要
break;
}
}
}else{
$id = getmypid();
echo "Child process,pid {$id}\n";
sleep(2);
}
该示例里只有一个子进程,看不出来非阻塞的好处,我们修改一下:
<?php
$child_pids = [];
for($i=0;$i<3; $i++){
$pid = pcntl_fork();
if($pid == -1){
exit("fork fail");
}elseif($pid){
$child_pids[] = $pid;
$id = getmypid();
echo time()." Parent process,pid {$id}, child pid {$pid}\n";
}else{
$id = getmypid();
$rand = rand(1,3);
echo time()." Child process,pid {$id},sleep $rand\n";
sleep($rand); //#1 故意设置时间不一样
exit();//#2 子进程需要exit,防止子进程也进入for循环
}
}
while(count($child_pids)){
foreach ($child_pids as $key => $pid) {
// $res = pcntl_wait($status, WNOHANG);
$res = pcntl_waitpid($pid, $status, WNOHANG);//#3
if ($res == -1 || $res > 0){
echo time()." Child process exit,pid {$pid}\n";
unset($child_pids[$key]);
}else{
// echo time()." Wait End,pid {$pid}\n"; //#4
}
}
}
#3
处首先先去掉WNOHANG
参数,运行:
$ php fork.1.php
1528637334 Parent process,pid 6600, child pid 6601
1528637334 Child process,pid 6601,sleep 2
1528637334 Parent process,pid 6600, child pid 6602
1528637334 Child process,pid 6602,sleep 2
1528637334 Parent process,pid 6600, child pid 6603
1528637334 Child process,pid 6603,sleep 1
1528637336 Child process exit,pid 6601
1528637336 Child process exit,pid 6602
1528637336 Child process exit,pid 6603
我们看到,6603号进程运行时间最短,但是是最后回收。我们再加上WNOHANG
参数,运行:
$ php fork.1.php
1528637511 Parent process,pid 6695, child pid 6696
1528637511 Child process,pid 6696,sleep 2
1528637511 Parent process,pid 6695, child pid 6697
1528637511 Child process,pid 6697,sleep 1
1528637511 Parent process,pid 6695, child pid 6698
1528637511 Child process,pid 6698,sleep 3
1528637512 Child process exit,pid 6697
1528637513 Child process exit,pid 6696
1528637514 Child process exit,pid 6698
6697进程最先回收!说明确实是异步非阻塞的。感兴趣的朋友还可以开启#4
处代码,未使用WNOHANG
参数的时候,里面的代码是不会运行的。
注意:#2
处需要注意子进程需要exit,防止子进程也进入for循环。如果没有exit()
,最终创建的子进程不只3个。
检测status函数
在 pcntl_wait
和pcntl_waitpid
两个函数中的$status
中存了子进程的状态信息,这个参数可以用于 pcntl_wifexited
、pcntl_wifstopped
、pcntl_wifsignaled
、pcntl_wexitstatus
、 pcntl_wtermsig
、pcntl_wstopsig
、pcntl_waitpid
这些函数。
代码片段:
while(1){
$res = pcntl_wait($status);
if ($res == -1 || $res > 0){
if(!pcntl_wifexited($status)){
//进程非正常退出
echo "service exit unusally; pid is $pid\n";
}else{
//获取进程终端的退出状态码;
$code = pcntl_wexitstatus($status);
echo "service exit code: $code;pid is $pid \n";
}
if(pcntl_wifsignaled($status)){
//不是通过接受信号中断
echo "service term not by signal;pid is $pid \n";
}else{
$signal = pcntl_wtermsig($status);
echo "service term by signal $signal;pid is $pid\n";
}
if(pcntl_wifstopped($status)){
echo "service stop not unusally;pid is $pid \n";
}else{
$signal = pcntl_wstopsig($status);
echo "service stop by signal $signal;pid is $pid\n";
}
break;
}
PHP多进程系列笔(转)的更多相关文章
- PHP多进程系列笔记(一)
本系列文章将向大家讲解pcntl_*系列函数,从而更深入的理解进程相关知识. PCNTL在PHP中进程控制支持默认是关闭的.您需要使用 --enable-pcntl 配置选项重新编译PHP的 CGI或 ...
- PHP多进程系列笔记(五)
前面几节都是讲解pcntl扩展实现的多进程程序.本节给大家介绍swoole扩展的swoole_process模块. swoole多进程 swoole_process 是swoole提供的进程管理模块, ...
- PHP多进程系列笔记(三)
本节讲解几个多进程的实例. 多进程实例 Master-Worker结构 下面例子实现了简单的多进程管理: 支持设置最大子进程数 Master-Worker结构:Worker挂掉,Master进程会重新 ...
- PHP多进程系列笔记(二)
上一篇文章讲解了pcntl_fork和pcntl_wait两个函数的使用,本篇继续讲解PHP多进程相关新知识. 僵尸(zombie)进程 这里说下僵尸进程: 僵尸进程是指的父进程已经退出,而该进程de ...
- PHP多进程系列笔记(四)
本节主要讲解Posix常用函数和进程池的概念,也会涉及到守护进程的知识.本节难度较低. Posix常用函数 posix_kill 向指定pid进程发送信号.成功时返回 TRUE , 或者在失败时返回 ...
- PHP Socket 编程进阶指南
学习准备 Linux 或者 Mac 环境: 安装有 Sockets 扩展: 了解 TCP/IP 协议. socket函数只是PHP扩展的一部分,编译PHP时必须在配置中添加 --enable-sock ...
- 系列3|走进Node.js之多进程模型
文:正龙(沪江网校Web前端工程师) 本文原创,转载请注明作者及出处 之前的文章"走进Node.js之HTTP实现分析"中,大家已经了解 Node.js 是如何处理 HTTP 请求 ...
- python量化分析系列之---5行代码实现1秒内获取一次所有股票的实时分笔数据
python量化分析系列之---5行代码实现1秒内获取一次所有股票的实时分笔数据 最近工作太忙了,有一个星期没有更新文章了,本来这一期打算分享一些对龙虎榜数据的分析结果的,现在还没有把数据内的价值很好 ...
- ACM_招新笔试题系列——买包子
招新笔试题系列——买包子 Time Limit: 2000/1000ms (Java/Others) Problem Description: 小华刚到大学,一天早上她替她室友买早餐,一共要N个包子. ...
随机推荐
- HTML 文件路径
文件路径描述了网站文件夹结构中某个文件的位置. 文件路径会在链接外部文件时被用到: 网页 图像 样式表 JavaScript 绝对文件路径 绝对文件路径是指向一个因特网文件的完整 URL: 实例 &l ...
- 前端知识体系-NodeJS相关】NodeJS基础知识全面总结
NodeJS基础知识 1. Node的全局对象和全局变量 1.1 全局对象:所有模块都可以调用的 global:表示Node所在的全局环境,类似于浏览器的window对象. process:该对象表示 ...
- angularjs 实现猜数字大小的功能
<body ng-app="myapp" ng-controller="myCtrl"> <h2>猜一猜,多大值?(1-1000)< ...
- bps和pps
bps,比特率指的是每秒传输比特数 在实际所说的1M带宽的意思是1Mbps(是兆比特每秒Mbps不是兆字节每秒MBps) pps(数据包每秒),常用的网络吞吐率的单位(即每秒发送多少个分组数据包),网 ...
- 机器学习(1)——K近邻算法
KNN的函数写法 import numpy as np from math import sqrt from collections import Counter def KNN_classify(k ...
- Druid-类图-属性表
所属文章:池化技术(一)Druid是如何管理数据库连接的? 本篇为「工具人」文章,建议直接用「ctrl+f」进行查找属性.方法.类名,快速了解其含义和所属类. 主要流程里主要涉及到的类名称.类属性.类 ...
- python中可变与不可变类型的全局变量
python中的不可变类型的全局变量如int a=1,str b='hello', 若需要修改必须加global申明, 而全局变量是可变类型的,如list, dict ,则直接修改list.app ...
- 详解扩展欧几里得算法(扩展GCD)
浅谈扩展欧几里得(扩展GCD)算法 本篇随笔讲解信息学奥林匹克竞赛中数论部分的扩展欧几里得算法.为了更好的阅读本篇随笔,读者最好拥有不低于初中二年级(这是经过慎重考虑所评定的等级)的数学素养.并且已经 ...
- ASP.NET CORE HOW TO ADD "ACCESS-CONTROL-EXPOSE-HEADERS" HEADERS?
services.AddCors(options => { options.AddPolicy("AnotherPolicy" ...
- Codechef October Challenge 2019 Division 1
Preface 这次CC难度较上两场升高了许多,后面两题都只能借着曲明姐姐和jz姐姐的仙气来做 值得一提的是原来的F大概需要大力分类讨论,结果我写了一大半题目就因为原题被ban了233 最后勉强涨了近 ...