zoukankan      html  css  js  c++  java
  • php多进程编程实现与优化

    PHP多进程API

    创建子进程

    @params void
    @returns int
    int pcntl_fork(void)
    成功时,在父进程执行线程内返回产生的子进程PID,在子进程执行线程内返回0,失败时,在父进程上下文返回-1,不会创建子进程,并且会引发一个php错误
    

    获取当前进程id

    @params void
    @returns int
    int posix_getpid(void)
    返回进程id,类型为整型
    

    父进程等待子进程退出

    @params $status
    @params $option
    @return bool
    int pcntl_wait(int &$status[,int $options=0])
    该函数等同于以-1作为参数pid的值并且没有options参数来调用pcntl_waitpid()的函数
    

    进程退出状态

    @params $status
    @return bool
    bool pcntl_wifexited(int $status)
    

    进程退出码

    @params $status
    @return int
    int pcntl_wexitstatus(int $status)
    

    简单PHP多进程示例

    function process_execute($input) {
            $pid = pcntl_fork(); //创建子进程
            if ($pid == 0) {//子进程
                    $pid = posix_getpid();
                    echo "* Process {$pid} was created, and Executed:
    
    ";
                    eval($input); //解析命令
                    exit;
            } else {//主进程
                    $pid = pcntl_wait($status, WUNTRACED); //取得子进程结束状态
                    if (pcntl_wifexited($status)) {
                            echo "
    
    * Sub process: {$pid} exited with {$status}";
                    }
            }
    

    通过调用php创建子进程接口完成一个子进程的创建,pcntl_fork返回值为0证明进入到子进程内,非0则进入到父进程内部,-1则父进程创建子进程失败。

    多个子进程初级版本示例

    foreach ($clusterList as $key=>$value) {
                $pid = pcntl_fork();//创建子进程
                if($pid == 0) {//子进程
                    //do something
                } else if($pid == -1) {
                    //fork error occured
                } else {
                    pcntl_wait($status);
                }
    
            }
    

    该实现方式主要逻辑为循环创建一个子进程,并且父进程等待子进程完成退出后,再继续创建下一个子进程
    缺点:无法真正体现多进程,实际上时串行的创建子进程

    多个子进程优化版本示例

    foreach ($clusterList as $key=>$value) {
                $pid = pcntl_fork();//创建子进程
                if($pid == 0) {//子进程
                    //do something
                } else if($pid == -1) {
                    return false;
                }
            }
            for (;;) {
                $ret = pcntl_waitpid(-1,$status,WNOHANG);
                if ($ret == -1) {
                    // error occured 
                } else if ($ret == 0) {
                    //all child are existed
                    break;
                } else {
                    //check sub process exit status
                    $extFlag = pcntl_wifexited($status);
                    if(!$extFlag){
                        //exited unnormally
                    }else {
                        $extCode = pcntl_wexitstatus($status);
                        //exited normally
                    }
                }
            }
    

    该逻辑通过for循环不断获取子进程的退出状态,直到所有的子进程都退出,真正实现多进程处理。

  • 相关阅读:
    打开一个网页,以html代码保存于txt文件中
    用C查看系统任务管理器中运行的程序
    常见两种LINK错误
    怎么把下载的dll和def生成lib,以用于编程
    建立一个不能打开的文件(占坑)C语言高级API调用
    [转]软件版本命名格式
    回调函数编写和注重点
    ubuntu linux mysql 开发模式与连接编译
    创建一个进程和两个管道,实现与进程通信
    hdoj 1115 Lifting the Stone (求多边形重心)
  • 原文地址:https://www.cnblogs.com/jingliming/p/9100309.html
Copyright © 2011-2022 走看看