VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > PHP >
  • php安装pcntl扩展实现多进程

pcntl中的php必须要安装pcntl才可以实现多线程了,在网上找到许多的关于pcntl安装教程,下面整理了一篇比较完整的关于php pcntl安装与使用方法.

pcntl中php实现多进程必须要安装的扩展,现将扩展安装步骤写在下面。

一、两种安装方式

1、重新编译PHP的后面configrue提示加上?enable-pcntl。

2、不重新编译php,直接编译安装pcntl扩展。

# cd /usr/local/src/php-5.2.6/ext/pcntl

# /usr/local/php/bin/phpize

# ./configure ?with-php-config=/usr/local/php/bin/php-config

# make && make install

然后将,pcntl.so 加到php.ini中就可以了,使用php -m查看模块命令可以查看已安装的模块。

二、实例

  1. for($x = 1;$x<= 2;$x++){ 
  2.  
  3.         $pid[$x] = pcntl_fork(); 
  4.  
  5.         if ($pid[$x] == -1) { 
  6.  
  7.                 die("could not fork"); 
  8.  
  9.         } elseif ($pid[$x]) { 
  10.  
  11.                 echo "Parent: create ".$pid[$x]."n"
  12.  
  13.         } else { 
  14.  
  15.                         echo "fork ".getmypid()." start:n"
  16.  
  17.                         for($i = 0;$i<10;$i++){ 
  18.  
  19.                                 echo $x.": ".$i."n"
  20.  
  21.                                 sleep(1); 
  22.  
  23.                         } 
  24.  
  25.                 exit
  26.  
  27.         } 
  28.  

本文实例讲述了PHP的pcntl多进程用法。分享给大家供大家参考。具体分析如下:

PHP使用PCNTL系列的函数也能做到多进程处理一个事务。比如我需要从数据库中获取80w条的数据,再做一系列后续的处理,这个时候,用单进程?你可以等到明年今天了。所以应该使用pcntl函数了。

假设我想要启动20个进程,将1-80w的数据分成20份来做,主进程等待所有子进程都结束了才退出:

  1. $max = 800000; 
  2.  
  3. $workers = 20; 
  4.  
  5. $pids = array(); 
  6.  
  7. for($i = 0; $i < $workers$i++){ 
  8.  
  9.     $pids[$i] = pcntl_fork(); 
  10.  
  11.     switch ($pids[$i]) { 
  12.  
  13.         case -1: 
  14.  
  15.             echo "fork error : {$i} \r\n"
  16.  
  17.             exit
  18.  
  19.         case 0: 
  20.  
  21.             $param = array
  22.  
  23.                 'lastid' => $max / $workers * $i
  24.  
  25.                 'maxid' => $max / $workers * ($i+1), 
  26.  
  27.             ); 
  28.  
  29.             $this->executeWorker($input$output$param); 
  30.  
  31.             exit
  32.  
  33.         default
  34.  
  35.             break
  36.  
  37.     } 
  38.  
  39.  
  40. foreach ($pids as $i => $pid) { 
  41.  
  42.     if($pid) { 
  43.  
  44.         pcntl_waitpid($pid$status); 
  45.  
  46.     } 
  47.  

这里当pcntl_fork出来以后,会返回一个pid值,这个pid在子进程中看是0,在父进程中看是子进程的pid(>0),如果pid为-1说明fork出错了。

使用一个$pids数组就可以让主进程等候所有进程完结之后再结束了

例子测试

pcntl的扩展的安装就不说了,很简单,这里结合实例说一下pcntl_fork的运行方式

  1. <?php 
  2.  
  3.   echo $pid= pcntl_fork(); 
  4.  
  5.   if ($pid == -1) {  
  6.  
  7.      die('could not fork');  
  8.  
  9.   } else if (!$pid) { 
  10.  
  11.    //这里是子进程 
  12.  
  13.   echo '-'
  14.  
  15.   exit(); 
  16.  
  17.   }else
  18.  
  19.   //这里是父进程 
  20.  
  21.  echo 'A'
  22.  
  23.  echo 'B'
  24.  
  25.   } 

我用cli运行多次的结果是

$pid为0说明是子进程,因为pcntl_fork()的作用就是当程序运行到这里,就试图去创建一个子进程,如果创建成果,那么返回当前进程的子进程id,0表示当前进程没有子进程,那么其进程本身不就是子进程了,而如果有进程id,表示其有子进程,那么其不就是父进程。就好像是从这个位置其,下面的所有代码被复制到了另一个进程中执行一样,产生了一个子进程。但是从上面多次执行结果看,大部分情况下是先执行子进程的,这和我之前网上查的资料有些出入,而且这个顺序也不是固定的。但是我们发现倒数第三条,是先执行父进程,后执行的子进程,这时候子进程是有一定风险的,虽然子进程不会死,因为他会过继到1进程,但如果要等待子进程执行完可以这样:

  1. <?php 
  2.   echo $pid= pcntl_fork(); 
  3.   if ($pid == -1) {  
  4.      die('could not fork');  
  5.   } else if (!$pid) { 
  6.    //这里是子进程 
  7.   echo '-'
  8.   exit(); 
  9.   }else
  10.   //这里是父进程 
  11.  echo 'A'
  12.   pcntl_wait($status);//父进程执行到这里等等子进程执行完再执行 
  13.  echo 'B'
  14.   } 
  15. ?> 

运行结果:

  1. 0-31843AB[root@client 60.test.com]# php index.php  
  2. 0-31845AB[root@client 60.test.com]# php index.php  
  3. 0-31847AB[root@client 60.test.com]# php index.php  
  4. 0-31849AB[root@client 60.test.com]# php index.php  
  5. 0-31851AB[root@client 60.test.com]# php index.php  
  6. 0-31853AB[root@client 60.test.com]# php index.php  
  7. 0-31855AB[root@client 60.test.com]# php index.php  
  8. 31857A0-B[root@client 60.test.com]# php index.php  
  9. 0-31859AB[root@client 60.test.com]# php index.php  
  10. 0-31861AB[root@client 60.test.com]# php index.php  
  11. 0-31863AB[root@client 60.test.com]# php index.php  
  12. 0-31865AB[root@client 60.test.com]# php index.php  
  13. 0-31867AB[root@client 60.test.com]# php index.php  
  14. 0-31869AB[root@client 60.test.com]# php index.php  
  15. 0-31871AB[root@client 60.test.com]# php index.php  
  16. 0-31873AB[root@client 60.test.com]# php index.php  
  17. 0-31875AB[root@client 60.test.com]# php index.php  
  18. 0-31877AB[root@client 60.test.com]# php index.php  
  19. 0-31879AB[root@client 60.test.com]# php index.php  
  20. 0-31881AB[root@client 60.test.com]# php index.php  
  21. 0-31883AB[root@client 60.test.com]# php index.php 

就是我们想要的结果了,其他都没有问题,因为都是先执行的子进程,而这条是先执行了父进程,走到输出A就等子进程执行完才执行的下面的。所以这就是

ntl_wait($status) 的作用。

好了,基本就是这些了。至于进程间或线程间的通信,可以利用类似共享内存变量等的来做。具体情况具体对待吧。

出处:http://www.phpfensi.com/php/20160825/10570.html

 


相关教程