加入收藏 | 设为首页 | 会员中心 | 我要投稿 PHP编程网 - 黄冈站长网 (http://www.0713zz.com/)- 数据应用、建站、人体识别、智能机器人、语音技术!
当前位置: 首页 > 站长学院 > PHP教程 > 正文

php安装pcntl扩展实现多进程

发布时间:2022-06-20 18:08:47 所属栏目:PHP教程 来源:互联网
导读:pcntl中的php必须要安装pcntl才可以实现多线程了,在网上找到许多的关于pcntl安装教程,下面整理了一篇比较完整的关于php pcntl安装与使用方法. pcntl中php实现多进程必须要安装的扩展,现将扩展安装步骤写在下面。 一、两种安装方式 1、重新编译PHP的后面co
  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查看模块命令可以查看已安装的模块。
 
  二、实例
 
  for($x = 1;$x<= 2;$x++){
  
          $pid[$x] = pcntl_fork();
  
          if ($pid[$x] == -1) {
  
                  die("could not fork");
  
          } elseif ($pid[$x]) {
  
                  echo "Parent: create ".$pid[$x]."n";
  
          } else {
  
                          echo "fork ".getmypid()." start:n";
  
                          for($i = 0;$i<10;$i++){
  
                                  echo $x.": ".$i."n";
  
                                  sleep(1);
                          }
  
                 exit;
         }
  
  }
  本文实例讲述了PHP的pcntl多进程用法。分享给大家供大家参考。具体分析如下:
 
  PHP使用PCNTL系列的函数也能做到多进程处理一个事务。比如我需要从数据库中获取80w条的数据,再做一系列后续的处理,这个时候,用单进程?你可以等到明年今天了。所以应该使用pcntl函数了。
 
  假设我想要启动20个进程,将1-80w的数据分成20份来做,主进程等待所有子进程都结束了才退出:
 
  $max = 800000;
  
  $workers = 20;
  
  $pids = array();
  
  for($i = 0; $i < $workers; $i++){
  
      $pids[$i] = pcntl_fork();
  
      switch ($pids[$i]) {
  
          case -1:
  
              echo "fork error : {$i} rn";
  
              exit;
  
          case 0:
  
              $param = array(
  
                  'lastid' => $max / $workers * $i,
  
                  'maxid' => $max / $workers * ($i+1),
  
              );
  
              $this->executeWorker($input, $output, $param);
  
              exit;
  
          default:
  
              break; 
      }
  
  }
  
  foreach ($pids as $i => $pid) {
  
      if($pid) {
  
          pcntl_waitpid($pid, $status);
  
      }
  
  }
  这里当pcntl_fork出来以后,会返回一个pid值,这个pid在子进程中看是0,在父进程中看是子进程的pid(>0),如果pid为-1说明fork出错了。
 
  使用一个$pids数组就可以让主进程等候所有进程完结之后再结束了
 
  例子测试
 
  pcntl的扩展的安装就不说了,很简单,这里结合实例说一下pcntl_fork的运行方式
 
  <?php
  
    echo $pid= pcntl_fork();
  
    if ($pid == -1) {  
  
       die('could not fork');  
  
    } else if (!$pid) {
  
     //这里是子进程
  
    echo '-';
  
    exit();
  
    }else{
  
    //这里是父进程
  
   echo 'A';
  
   echo 'B';
  
    }
  我用cli运行多次的结果是
 
  $pid为0说明是子进程,因为pcntl_fork()的作用就是当程序运行到这里,就试图去创建一个子进程,如果创建成果,那么返回当前进程的子进程id,0表示当前进程没有子进程,那么其进程本身不就是子进程了,而如果有进程id,表示其有子进程,那么其不就是父进程。就好像是从这个位置其,下面的所有代码被复制到了另一个进程中执行一样,产生了一个子进程。但是从上面多次执行结果看,大部分情况下是先执行子进程的,这和我之前网上查的资料有些出入,而且这个顺序也不是固定的。但是我们发现倒数第三条,是先执行父进程,后执行的子进程,这时候子进程是有一定风险的,虽然子进程不会死,因为他会过继到1进程,但如果要等待子进程执行完可以这样:
 
  <?php
    echo $pid= pcntl_fork();
    if ($pid == -1) {  
       die('could not fork');  
    } else if (!$pid) {
     //这里是子进程
    echo '-';
    exit();
    }else{
    //这里是父进程
   echo 'A';
    pcntl_wait($status);//父进程执行到这里等等子进程执行完再执行
   echo 'B';
    }
  ?>
  运行结果:
 
  0-31843AB[root@client 60.test.com]# php index.php  
  0-31845AB[root@client 60.test.com]# php index.php  
  0-31847AB[root@client 60.test.com]# php index.php  
  0-31849AB[root@client 60.test.com]# php index.php  
  0-31851AB[root@client 60.test.com]# php index.php  
  0-31853AB[root@client 60.test.com]# php index.php  
  0-31855AB[root@client 60.test.com]# php index.php  
  31857A0-B[root@client 60.test.com]# php index.php  
  0-31859AB[root@client 60.test.com]# php index.php  
  0-31861AB[root@client 60.test.com]# php index.php  
  0-31863AB[root@client 60.test.com]# php index.php  
  0-31865AB[root@client 60.test.com]# php index.php  
  0-31867AB[root@client 60.test.com]# php index.php  
  0-31869AB[root@client 60.test.com]# php index.php  
  0-31871AB[root@client 60.test.com]# php index.php  
  0-31873AB[root@client 60.test.com]# php index.php  
  0-31875AB[root@client 60.test.com]# php index.php  
  0-31877AB[root@client 60.test.com]# php index.php  
  0-31879AB[root@client 60.test.com]# php index.php  
  0-31881AB[root@client 60.test.com]# php index.php  
  0-31883AB[root@client 60.test.com]# php index.php
  
      就是我们想要的结果了,其他都没有问题,因为都是先执行的子进程,而这条是先执行了父进程,走到输出A就等子进程执行完才执行的下面的。所以这就是
 
  ntl_wait($status) 的作用。
 
  好了,基本就是这些了。至于进程间或线程间的通信,可以利用类似共享内存变量等的来做。具体情况具体对待吧。
 
 

(编辑:PHP编程网 - 黄冈站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    热点阅读