前言
在服务器跑脚本时,避免不了一些耗时任务,使用多进程是必不可少的。而在 PHP5.5 之后,PHP 开始加入了多进程元素,以满足开发需求。
注意
- 实现多进程需要开启的扩展:pcntl、 posix。
- Windows 环境下不支持 PHP 的多进程编程,本文主要在 Linux 环境下开发测试
一张简单结构图
主要功能
主要参考官方文档 进程控制
- pcntl_fork:创建多进程,调用后会返回两条进程的pid,0 为子进程,大于 0 为父进程(父进程得到子进程的 id,所以大于 0),-1为创建失败
$pid = $pcntlInstall ? pcntl_fork() : 0;
if ($pid == -1) {
//fork失败
} elseif ($pid > 0) {
//父进程
......
} elseif ($pid == 0) {
//子进程
......
}
- pcntl_signal: 注册一个信号处理回调函数,可以捕获子进程结束时发出的信号
/配合pcntl_signal使用
declare (ticks = 1);
//当子进程退出时,会触发该函数,当前子进程数-1
pcntl_signal(SIGCHLD, function ($signo) {
switch ($signo) {
case SIGCHLD:
echo $curChildPro . 'SIGCHLD', PHP_EOL;
$curChildPro--;
break;
}
});
- pcntl_wait: 用来暂停父进程,等待子进程退出
一个多进程的例子
该例子主要介绍了如何控制 同一时刻 进程 并发 的数量
$curChildPro = 0;
$maxChildPro = 5; // 同一时刻最多 5 个进程
//配合pcntl_signal使用
declare (ticks = 1);
//当子进程退出时,会触发该函数,当前子进程数-1
pcntl_signal(SIGCHLD, function ($signo) {
global $curChildPro;
switch ($signo) {
case SIGCHLD:
echo $curChildPro . 'SIGCHLD', PHP_EOL;
$curChildPro--;
break;
}
});
$index = 0;
while ($index < 10) {
$index ++;
$curChildPro++;
echo "-------- current process" . $curChildPro . "--------\r\n";
$pid = $pcntlInstall ? pcntl_fork() : 0;
if ($pid == -1) {
//fork失败
} elseif ($pid > 0) {
//达到上限时父进程阻塞等待任一子进程退出后while循环继续
if ($curChildPro >= $maxChildPro) {
pcntl_wait($status);
}
} elseif ($pid == 0) {
//子进程 执行一些操作
......
exit(); // 需要退出,避免产生僵尸进程
}
}