msgbartop
PHP语言, PHP扩展, Zend引擎相关的研究,技术,新闻分享 – 左手代码 右手诗
msgbarbottom

11 Jun 09 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开发中使用多进程的念头吧!


分享到:



Related Posts:

Tags: , , , ,

29 Responses to “PHP CLI模式下的多进程应用”

  1. PHP WES 可扩展 任务系统 (分布采集系统 + 采集器集群(单机百万级任务)+分布式文件系统) – (●--●) Hello.My Weicot |

    [...] 整合 Selenium2 + PhantomJS Jun 09 PHP CLI模式下的多进程应用Linux进程间通信 – 信号(signal)机制 [...]

  2. oracle chinois |

    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 |

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

  4. 横竖勾乘二 |

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

  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 |

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

  7. rainsun |

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

  8. Anonymous |

    你好,我从shell运行你的代码,提示错误
    PHP Fatal error: Call to undefined function readline()

    网上说是编译php的时候要加上 –with-readline ,但是还是没用,请教啊

  9. PHP CLI模式下的多进程应用 | 顾翔的冰冷小站 |

    [...] 代码如下, 我加了注释, 如果有不懂的地方, 可以翻阅手册相关函数, 或者回复留言. [...]

  10. PHP CLI模式下的多进程应用 |

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

  11. YuBing |

    @laruence

    今天又翻到这个帖子,顺带去确认了一下。

    CentOS里面是/usr/bin/env和/bin/env都存在,前者是一个软连接。

    不过我的Mac下面只有前者,没有/bin/env。

    手头没有别的操作系统,不过暂时来看,/usr/bin/env 稍稍通用一点点:P

  12. 雪候鸟 |

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

  13. YuBing |

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

  14. php-cli模式学习(PHP命令行模式) | PHP10086博客网 |

    [...] http://www.laruence.com/2009/06/11/930.html [...]

  15. Anonymous |

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

  16. 伴夜 |

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

  17. evoup |

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

  18. 拓荒.net |

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

  19. PHP CLI模式下的多进程应用 | Jim的blog |

    [...] 代码如下, 我加了注释, 如果有不懂的地方, 可以翻阅手册相关函数, 或者回复留言. [...]

  20. PHP CLI模式下的多进程应用 | 万维网黑客联盟 |

    [...] 本文地址: http://www.laruence.com/2009/06/11/930.html [...]

  21. Mpass – PHP做Socket服务的解决方案 | 天道酬勤 |

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

  22. Mpass – PHP做Socket服务的解决方案 » NoNZero's Blog |

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

  23. PHP CLI模式下的多进程应用 » NoNZero's Blog |

    [...] 代码如下, 我加了注释, 如果有不懂的地方, 可以翻阅手册相关函数, 或者回复留言. [...]

  24. hfcorriez |

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

  25. Mpass – PHP做Socket服务的解决方案 | 风雪之隅 |

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

  26. jackyd |

    这篇介绍5.3GC的文章貌似不错:
    http://blog.csdn.net/phpkernel/archive/2010/07/14/5734743.aspx

  27. 雪候鸟 |

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

  28. zhoushine |

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

  29. 如何化妆 |

    很实用

Leave a Reply

*