下面要分享一段开启多进程的PHP代码,不多做解释,都在注释里面。
本文实例讲述了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开启多进程的方法,希望对你有所帮助。
点击查看更多内容
为 TA 点赞
评论
共同学习,写下你的评论
评论加载中...
作者其他优质文章
正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦