php开启多进程的方法
程序员文章站
2022-05-18 07:53:41
本文实例讲述了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开启多进程的方法。分享给大家供大家参考。具体实现方法如下:
复制代码 代码如下:
<?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程序设计有所帮助。
上一篇: IP负载均衡