php多进程实现

PHP有一组进程控制函数(编译时需要–enable-pcntl与posix扩展),使得php能在nginx系统中实现跟c一样的创建子进程、使用exec函数执行程序、处理信号等功能。

CentOS 6 下yum安装php的,默认是不安装pcntl的,因此需要单独编译安装,首先下载对应版本的php,解压后




[plain] view plain copy

 

 print?

  1. cd php-version/ext/pcntl  
  2. phpize  
  3. ./configure && make && make install  
  4. cp /usr/lib/php/modules/pcntl.so /usr/lib64/php/modules/pcntl.so  
  5. echo "extension=pcntl.so" >> /etc/php.ini  
  6. /etc/init.d/httpd restart  


方便极了。


 

下面是示例代码:




[php] view plain copy

 

 print?

  1. <?php  
  2. header('content-type:text/html;charset=utf-8' );  
  3.   
  4. // 必须加载扩展  
  5. if (!function_exists("pcntl_fork")) {  
  6.     die("pcntl extention is must !");  
  7. }  
  8. //总进程的数量  
  9. $totals = 3;  
  10. // 执行的脚本数量  
  11. $cmdArr = array();  
  12. // 执行的脚本数量的数组  
  13. for ($i = 0; $i < $totals; $i++) {  
  14.     $cmdArr[] = array("path" => __DIR__ . "/run.php",  'pid' =>$i ,'total' =>$totals);  
  15. }  
  16.   
  17. /* 
  18. 展开:$cmdArr 
  19. Array 
  20.     [0] => Array 
  21.         ( 
  22.             [path] => /var/www/html/company/pcntl/run.php 
  23.             [pid] => 0 
  24.             [total] => 3 
  25.         ) 
  26.  
  27.     [1] => Array 
  28.         ( 
  29.             [path] => /var/www/html/company/pcntl/run.php 
  30.             [pid] => 1 
  31.             [total] => 3 
  32.         ) 
  33.  
  34.     [2] => Array 
  35.         ( 
  36.             [path] => /var/www/html/company/pcntl/run.php 
  37.             [pid] => 2 
  38.             [total] => 3 
  39.         ) 
  40.  
  41. */  
  42.   
  43. pcntl_signal(SIGCHLD, SIG_IGN); //如果父进程不关心子进程什么时候结束,子进程结束后,内核会回收。  
  44. foreach ($cmdArr  as   $cmd) {  
  45.     $pid = pcntl_fork();    //创建子进程  
  46.     //父进程和子进程都会执行下面代码  
  47.     if ($pid == -1) {  
  48.         //错误处理:创建子进程失败时返回-1.  
  49.         die('could not fork');  
  50.     } else if ($pid) {  
  51.         //父进程会得到子进程号,所以这里是父进程执行的逻辑  
  52.         //如果不需要阻塞进程,而又想得到子进程的退出状态,则可以注释掉pcntl_wait($status)语句,或写成:  
  53.         pcntl_wait($status,WNOHANG); //等待子进程中断,防止子进程成为僵尸进程。  
  54.     } else {  
  55.         //子进程得到的$pid为0, 所以这里是子进程执行的逻辑。  
  56.         $path   = $cmd["path"];  
  57.         $pid = $cmd['pid'] ;  
  58.         $total = $cmd['total'] ;  
  59.         echo exec("/usr/bin/php {$path} {$pid} {$total}")."\n";  
  60.         exit(0) ;  
  61.     }  
  62. }  
  63. ?>