当前位置: 首页 > 编程笔记 >

php开启多进程的方法

万俟玉书
2023-03-14
本文向大家介绍php开启多进程的方法,包括了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程序设计有所帮助。

 类似资料:
  • 本文向大家介绍深入探究PHP的多进程编程方法,包括了深入探究PHP的多进程编程方法的使用技巧和注意事项,需要的朋友参考一下 子进程的创建 一般的子进程的写法是:     上边的代码如果创建子进程成功的话,系统就有了2个进程,一个为父进程,一个为子进程,子进程的id号为$pid。在系统运行到$pid = pcntl_fork();时,在这个地方进行分支,父子进程各自开始运行各自的程序代码。代码的运行

  • 干巴巴地叨逼叨了这么久,时候表演真正的技术了! 做个高端点儿的玩意吧,假如我们要做一个任务系统,这个系统可以在后台帮我们完成一大波(注意是一大波)数据的处理,那么我们自然想到,多开几个进程分开处理这些数据,同时我们不能执行了 php task.php 后终端挂起,万一一不小心关闭了终端都会导致任务失败,所以我们还要实现程序的 daemon 化。 首先,我们第一步就得将程序 daemon 化了!

  • 本文向大家介绍PHP多进程编程实例,包括了PHP多进程编程实例的使用技巧和注意事项,需要的朋友参考一下 羡慕火影忍者里鸣人的影分身么?没错,PHP程序是可以开动影分身的!想完成任务,又觉得一个进程太慢,那么,试试用多进程来搞吧。这篇文章将会介绍一下PHP多进程的基本需求,如何创建多进程以及基本的信号控制,暂时不会告诉你如何进行进程间通信和信息共享。 1. 准备 在动手之前,请确定你用的不是M$ W

  • 本文向大家介绍PHP exif扩展方法开启详解,包括了PHP exif扩展方法开启详解的使用技巧和注意事项,需要的朋友参考一下 服务器配置说明: 1.在php.ini文件中找到;extension=php_exif.dll,去掉前面的分号 2.在php.ini文件中找到;extension=php_mbstring.dll,去掉前面的分号,并将此行移动到extension=php_exif.dll

  • 本文向大家介绍Linux启动新进程的三种方法,包括了Linux启动新进程的三种方法的使用技巧和注意事项,需要的朋友参考一下 程序中,我们有时需要启动一个新的进程,来完成其他的工作。 下面介绍了三种实现方法,以及这三种方法之间的区别。 1.system函数-调用shell进程,开启新进程 system函数,是通过启动shell进程,然后执行shell命令进程。 原型: int system(cons

  • 本文向大家介绍C#在Unity游戏开发中进行多线程编程的方法,包括了C#在Unity游戏开发中进行多线程编程的方法的使用技巧和注意事项,需要的朋友参考一下 在这之前,有很多人在质疑Unity支不支持多线程,事实上Unity是支持多线程的。而提到多线程就要提到Unity非常常用的协程,然而协程并非真正的多线程。协程其实是等某个操作完成之后再执行后面的代码,或者说是控制代码在特定的时机执行。而多线程在