• 技术文章 >后端开发 >php教程

    php开启多进程的方法_PHP

    2016-05-31 13:16:42原创1006
    本文实例讲述了php开启多进程的方法。分享给大家供大家参考。具体实现方法如下:

    代码如下:


    <?php
    $IP='192.168.1.1';//Windows電腦的IP
    $Port='5900'; //VNC使用的Port
    $ServerPort='9999';//Linux Server對外使用的Port
    $RemoteSocket=false;//連線到VNC的Socket
    function SignalFunction($Signal){
    //這是主Process的訊息處理函數
    global $PID;//Child Process的PID
    switch ($Signal)
    {
    case SIGTRAP:
    case SIGTERM:
    //收到結束程式的Signal
    if($PID)
    {
    //送一個SIGTERM的訊號給Child告訴他趕快結束掉嘍
    posix_kill($PID,SIGTERM);
    //等待Child Process結束,避免zombie
    pcntl_wait($Status);
    }
    //關閉主Process開啟的Socket
    DestroySocket();
    exit(0); //結束主Process
    break;
    case SIGCHLD:
    /*
    當Child Process結束掉時,Child會送一個SIGCHLD訊號給Parrent
    當Parrent收到SIGCHLD,就知道Child Process已經結束嘍 ,該做一些
    結束的動作*/
    unset($PID); //將$PID清空,表示Child Process已經結束
    pcntl_wait($Status); //避免Zombie
    break;
    default:
    }
    }
    function ChildSignalFunction($Signal){
    //這是Child Process的訊息處理函數
    switch ($Signal)
    {
    case SIGTRAP:
    case SIGTERM:
    //Child Process收到結束的訊息
    DestroySocket(); //關閉Socket
    exit(0); //結束Child Process
    default:
    }
    }
    function ProcessSocket($ConnectedServerSocket){
    //Child Process Socket處理函數
    //$ConnectedServerSocket -> 外部連進來的Socket
    global $ServerSocket,$RemoteSocket,$IP,$Port;
    $ServerSocket=$ConnectedServerSocket;
    declare(ticks = 1); //這一行一定要加,不然沒辦法設定訊息處理函數。
    //設定訊息處理函數
    if(!pcntl_signal(SIGTERM, "ChildSignalFunction")) return;
    if(!pcntl_signal(SIGTRAP, "ChildSignalFunction")) return;
    //建立一個連線到VNC的Socket
    $RemoteSocket=socket_create(AF_INET, SOCK_STREAM,SOL_TCP);
    //連線到內部的VNC
    @$RemoteConnected=socket_connect($RemoteSocket,$IP,$Port);
    if(!$RemoteConnected) return; //無法連線到VNC 結束
    //將Socket的處理設為Nonblock,避免程式被Block住
    if(!socket_set_nonblock($RemoteSocket)) return;
    if(!socket_set_nonblock($ServerSocket)) return;
    while(true)
    {
    //這邊我們採用pooling的方式去取得資料
    $NoRecvData=false; //這個變數用來判別外部的連線是否有讀到資料
    $NoRemoteRecvData=false;//這個變數用來判別VNC連線是否有讀到資料
    @$RecvData=socket_read($ServerSocket,4096,PHP_BINARY_READ);
    //從外部連線讀取4096 bytes的資料
    @$RemoteRecvData=socket_read($RemoteSocket,4096,PHP_BINARY_READ);
    //從vnc連線連線讀取4096 bytes的資料
    if($RemoteRecvData==='')
    {
    //VNC連線中斷,該結束嘍
    echo"Remote Connection Close\n";
    return;
    }
    if($RemoteRecvData===false)
    {
    /*
    由於我們是採用nonblobk模式
    這裡的情況就是vnc連線沒有可供讀取的資料
    */
    $NoRemoteRecvData=true;
    //清除掉Last Errror
    socket_clear_error($RemoteSocket);
    }
    if($RecvData==='')
    {
    //外部連線中斷,該結束嘍
    echo"Client Connection Close\n";
    return;
    }
    if($RecvData===false)
    {
    /*
    由於我們是採用nonblobk模式
    這裡的情況就是外部連線沒有可供讀取的資料
    */
    $NoRecvData=true;
    //清除掉Last Errror
    socket_clear_error($ServerSocket);
    }
    if($NoRecvData&&$NoRemoteRecvData)
    {
    //如果外部連線以及VNC連線都沒有資料可以讀取時,
    //就讓程式睡個0.1秒,避免長期佔用CPU資源
    usleep(100000);
    //睡醒後,繼續作pooling的動作讀取socket
    continue;
    }
    //Recv Data
    if(!$NoRecvData)
    {
    //外部連線讀取到資料
    while(true)
    {
    //把外部連線讀到的資料,轉送到VNC連線上
    @$WriteLen=socket_write($RemoteSocket,$RecvData);
    if($WriteLen===false)
    {
    //由於網路傳輸的問題,目前暫時無法寫入資料
    //先睡個0.1秒再繼續嘗試。
    usleep(100000);
    continue;
    }
    if($WriteLen===0)
    {
    //遠端連線中斷,程式該結束了
    echo"Remote Write Connection Close\n";
    return;
    }
    //從外部連線讀取的資料,已經完全送給VNC連線時,中斷這個迴圈。
    if($WriteLen==strlen($RecvData)) break;
    //如果資料一次送不完就得拆成好幾次傳送,直到所有的資料全部送出為止
    $RecvData=substr($RecvData,$WriteLen);
    }
    }
    if(!$NoRemoteRecvData)
    {
    //這邊是從VNC連線讀取到的資料,再轉送回外部的連線
    //原理跟上面差不多不再贅述
    while(true)
    {
    @$WriteLen=socket_write($ServerSocket,$RemoteRecvData);
    if($WriteLen===false)
    {
    usleep(100000);
    continue;
    }
    if($WriteLen===0)
    {
    echo"Remote Write Connection Close\n";
    return;
    }
    if($WriteLen==strlen($RemoteRecvData)) break;
    $RemoteRecvData=substr($RemoteRecvData,$WriteLen);
    }
    }
    }
    }
    function DestroySocket(){
    //用來關閉已經開啟的Socket
    global$ServerSocket,$RemoteSocket;
    if($RemoteSocket)
    {
    //如果已經開啟VNC連線
    //在Close Socket前必須將Socket shutdown不然對方不知到你已經關閉連線了
    @socket_shutdown($RemoteSocket,2);
    socket_clear_error($RemoteSocket);
    //關閉Socket
    socket_close($RemoteSocket);
    }
    //關閉外部的連線
    @socket_shutdown($ServerSocket,2);
    socket_clear_error($ServerSocket);
    socket_close($ServerSocket);
    }
    //這裡是整個程式的開頭,程式從這邊開始執行
    //這裡首先執行一次fork
    $PID=pcntl_fork();
    if($PID==-1) die("could not fork");
    //如果$PID不為0表示這是Parrent Process
    //$PID就是Child Process
    //這是Parrent Process 自己結束掉,讓Child成為一個Daemon。
    if($PID) die("Daemon PID:$PID\n");
    //從這邊開始,就是Daemon模式在執行了
    //將目前的Process跟終端機脫離成為daemon模式
    if(!posix_setsid()) die("could not detach from terminal\n");
    //設定daemon 的訊息處理函數
    declare(ticks = 1);
    if(!pcntl_signal(SIGTERM, "SignalFunction")) die("Error!!!\n");
    if(!pcntl_signal(SIGTRAP, "SignalFunction")) die("Error!!!\n");
    if(!pcntl_signal(SIGCHLD, "SignalFunction")) die("Error!!!\n");
    //建立外部連線的Socket
    $ServerSocket=socket_create(AF_INET, SOCK_STREAM,SOL_TCP);
    //設定外部連線監聽的IP以及Port,IP欄位設0,表示經聽所有介面的IP
    if(!socket_bind($ServerSocket,0,$ServerPort)) die("Cannot Bind Socket!\n");
    //開始監聽Port
    if(!socket_listen($ServerSocket)) die("Cannot Listen!\n");
    //將Socket設為nonblock模式
    if(!socket_set_nonblock($ServerSocket)) die("Cannot Set Server Socket to Block!\n");
    //清空$PID變數,表示目前沒有任何的Child Process
    unset($PID);
    while(true)
    {
    //進入pooling模式,每隔1秒鐘就去檢查有沒有連線進來。
    sleep(1);
    //檢查有沒有連線進來
    @$ConnectedServerSocket=socket_accept($ServerSocket);
    if($ConnectedServerSocket!==false)
    {
    //有人連進來嘍
    //起始一個Child Process用來處理連線
    $PID=pcntl_fork();
    if($PID==-1) die("could not fork");
    if($PID) continue;//這是daemon process,繼續回去監聽。
    //這裡是Child Process開始
    //執行Socket裡函數
    ProcessSocket($ConnectedServerSocket);
    //處理完Socket後,結束掉Socket
    DestroySocket();
    //結束Child Process
    exit(0);
    }
    }

    希望本文所述对大家的php程序设计有所帮助。

    声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn核实处理。
    专题推荐:php 开启 多进程 方法
    上一篇:php中explode的负数limit用法分析_PHP 下一篇:初识laravel5_PHP
    PHP编程就业班

    相关文章推荐

    • 浅析PHP配置文件中的几种超时配置• 聊聊nginx平滑重启和FPM平滑重启• 归纳总结PHP对象基础• 请问疑难杂症 • 求这个网站的基本词挖掘功能的做法,求实现代码

    全部评论我要评论

  • 取消发布评论发送
  • 1/1

    PHP中文网