Press "Enter" to skip to content

PHP CLI模式下的多进程应用

PHP在很多时候不适合做常驻的SHELL进程, 他没有专门的gc例程, 也没有有效的内存管理途径. 所以如果用PHP做常驻SHELL, 你会经常被内存耗尽导致abort而unhappy.
而且, 如果输入数据非法, 而脚本没有检测, 导致abort, 也会让你很不开心.
那? 怎么办呢?
多进程....

为什么呢?

 优点:
    1. 使用多进程, 子进程结束以后, 内核会负责回收资源
    2. 使用多进程,子进程异常退出不会导致整个进程Thread退出. 父进程还有机会重建流程.
    3. 一个常驻主进程, 只负责任务分发, 逻辑更清楚.

Then, 怎么做呢?
接下来, 我们使用PHP提供的POSIX和Pcntl系列函数, 来实现一个PHP命令解析器, 主进程负责接受用户输入, 然后fork子进程执行, 并负责回显子进程的结束状态.
代码如下, 我加了注释, 如果有不懂的地方, 可以翻阅手册相关函数, 或者回复留言.

#!/bin/env php
<?php
/** A example denoted muti-process application in php
 * @filename fork.php
 * @touch date Wed 10 Jun 2009 10:25:51 PM CST
 * @author Laruence<laruence@baidu.com>
 * @license http://www.zend.com/license/3_0.txt   PHP License 3.0
 * @version 1.0.0
*/
/** 确保这个函数只能运行在SHELL中 */
if (substr(php_sapi_name(), 0, 3) !== 'cli') {
    die("This Programe can only be run in CLI mode");
}
/**  关闭最大执行时间限制, 在CLI模式下, 这个语句其实不必要 */
set_time_limit(0);
$pid  = posix_getpid(); //取得主进程ID
$user = posix_getlogin(); //取得用户名
echo <<<eod
USAGE: [command | expression]
input php code to execute by fork a new process
input quit to exit
        Shell Executor version 1.0.0 by laruence
EOD;
while (true) {
        $prompt = "\n{$user}$ ";
        $input  = readline($prompt);
        readline_add_history($input);
        if ($input == 'quit') {
			break;
		}
        process_execute($input . ';');
}
exit(0);
function process_execute($input) {
        $pid = pcntl_fork(); //创建子进程
        if ($pid == 0) {//子进程
                $pid = posix_getpid();
                echo "* Process {$pid} was created, and Executed:\n\n";
                eval($input); //解析命令
                exit;
        } else {//主进程
                $pid = pcntl_wait($status, WUNTRACED); //取得子进程结束状态
                if (pcntl_wifexited($status)) {
                        echo "\n\n* Sub process: {$pid} exited with {$status}";
                }
        }
}
   

但有一点, 我一定要提醒:

Process Control should not be enabled within a webserver environment and unexpected results may happen if any Process Control functions are used within a webserver environment.  --摘自PHP手册

也就是说, 打消你在PHP Web开发中使用多进程的念头吧!

38 Comments

  1. Love_PHP
    Love_PHP July 23, 2019

    开始的地方,EOD要大写,不然提示语法错误,我的linux php 7.1运行环境

  2. oracle chinois
    oracle chinois November 16, 2017

    Piece of writing writing is also a fun, if you be acqujainted with
    after that yyou can write or else it is difficult tto write.

  3. Thinklong
    Thinklong December 7, 2016

    CLI模式下最适合的就是写脚本,我开发了一套任务调度系统,之前开发过一套,但不是在Yaf下开发的,现在又在Yaf下重新开发了一套,可以实现多台机器动态调度、定时启动、信号控制、多进程、单进程、平滑重启、平滑reload(非主程序代码平滑更新)、守护进程。
    感谢鸟哥!

  4. 横竖勾乘二
    横竖勾乘二 July 31, 2015

    可惜看到晚了,今天多谢鸟哥微博里指导

  5. […] PHP5多层继承顺序的bug   HTTP1.0下HTTP_HOST为空  一个巧妙的分页方法 31 Dec 08 一个低概率的PHP Core dump 21 Feb 09 PHP字符串比较 26 May 09 PHP+Gtk实例(求24点)  PHP受locale影响的函数  PHP CLI模式下的多进程应用 […]

  6. yongbaolinux
    yongbaolinux November 7, 2014

    如果在webserver环境下不适合用进程控制 那么php的进程控制能够用在什么场合呢

  7. rainsun
    rainsun October 22, 2014

    大神 我想问一下我使用的是5.3
    为什么会没有函数呢posix_getpid()

  8. Anonymous
    Anonymous September 6, 2013

    你好,我从shell运行你的代码,提示错误
    PHP Fatal error: Call to undefined function readline()
    网上说是编译php的时候要加上 –with-readline ,但是还是没用,请教啊

  9. […] 接下来, 我们使用PHP提供的POSIX和Pcntl系列函数, 来实现一个PHP命令解析器, 主进程负责接受用户输入, 然后fork子进程执行, 并负责回显子进程的结束状态. 代码如下, 我加了注释, 如果有不懂的地方, 可以翻阅手册相关函数, 或者回复留言. […]

  10. YuBing
    YuBing November 28, 2012

    @laruence
    今天又翻到这个帖子,顺带去确认了一下。
    CentOS里面是/usr/bin/env和/bin/env都存在,前者是一个软连接。
    不过我的Mac下面只有前者,没有/bin/env。
    手头没有别的操作系统,不过暂时来看,/usr/bin/env 稍稍通用一点点:P

  11. 雪候鸟
    雪候鸟 September 25, 2012

    @YuBing linux/redhat,也就是我的开发机是/bin/env, 🙂

  12. YuBing
    YuBing September 23, 2012

    脚本第一行不该写 /bin/env,大部分linux系统或者mac应该都是/usr/bin/env吧?

  13. Anonymous
    Anonymous August 21, 2012

    可以不使用POSIX和Pcntl实现,既然要在服务器上命令行执行,就直接系统命令调用PHP运行.

  14. 伴夜
    伴夜 March 30, 2012

    请问 在windows下 PHP用CLI方式执行 在命令输入后会有很长一段时间的等待才会输出,比如 php -r “echo ‘123’;” 这句就大约需要3秒多,这是为什么呢?

  15. evoup
    evoup October 9, 2011

    php -r 可以吗?常驻shell啥意思

  16. 拓荒.net
    拓荒.net May 11, 2011

    我将源码拷贝运行测试,居然提示process_execute()未定义,我不得不把定一此函数的function process_execute()这段话放到调用之前,才能正常运行,不知道为什么……

  17. […] 我之前的文章PHP CLI模式下的多进程应用介绍过在PHP做服务的时候如何避免内存泄露, 所以基于此, 再加上PHP5以后的stream_socket_*系列API, 就完成了这一套框架. […]

  18. hfcorriez
    hfcorriez October 29, 2010

    学习,这个也只能在后台事务处理上用用。

  19. […] 我之前的文章PHP CLI模式下的多进程应用介绍过在PHP做服务的时候如何避免内存泄露, 所以基于此, 再加上PHP5以后的stream_socket_*系列API, 就完成了这一套框架. […]

  20. zhoushine
    zhoushine July 18, 2009

    我们这边有个类似系统就是用php去做的。怕gc有问题。子进程处理一定数量的请求后就退了重新fork出来。

    • 雪候鸟
      雪候鸟 July 20, 2009

      恩,PHP的GC,我搞到现在了还没完全搞清楚..释放的时机,释放的深度等等…

Comments are closed.