• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

php开启多进程的方法_PHP

php 搞代码 3年前 (2022-01-25) 11次浏览 已收录 0个评论

本文实例讲述了php开启多进程的方法。分享给大家供大家参考。具体实现方法如下:

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

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


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:php开启多进程的方法_PHP
喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址