php开启多进程的方法
来源: 阅读:1447 次 日期:2015-03-05 10:42:28
温馨提示: 小编为您整理了“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程序设计有所帮助。

更多信息请查看IT技术专栏

更多信息请查看网络编程
手机网站地址:php开启多进程的方法
由于各方面情况的不断调整与变化, 提供的所有考试信息和咨询回复仅供参考,敬请考生以权威部门公布的正式信息和咨询为准!

2025国考·省考课程试听报名

  • 报班类型
  • 姓名
  • 手机号
  • 验证码
关于我们 | 联系我们 | 人才招聘 | 网站声明 | 网站帮助 | 非正式的简要咨询 | 简要咨询须知 | 加入群交流 | 手机站点 | 投诉建议
工业和信息化部备案号:滇ICP备2023014141号-1 云南省教育厅备案号:云教ICP备0901021 滇公网安备53010202001879号 人力资源服务许可证:(云)人服证字(2023)第0102001523号
云南网警备案专用图标
联系电话:0871-65317125(9:00—18:00) 获取招聘考试信息及咨询关注公众号:
咨询QQ:526150442(9:00—18:00)版权所有:
云南网警报警专用图标
Baidu
map