Home >php教程 >PHP源码 >php安装pcntl扩展实现多进程

php安装pcntl扩展实现多进程

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOriginal
2016-06-08 17:19:561798browse

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

<script>ec(2);</script>

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} \r\n";

            exit;

        case 0:

            $param = array(

                &#39;lastid&#39; => $max / $workers * $i,

                &#39;maxid&#39; => $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(&#39;could not fork&#39;); 

  } else if (!$pid) {

   //这里是子进程

  echo &#39;-&#39;;

  exit();

  }else{

  //这里是父进程

 echo &#39;A&#39;;

 

 echo &#39;B&#39;;

  }

我用cli运行多次的结果是

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

<?php
  echo $pid= pcntl_fork();
  if ($pid == -1) { 
     die(&#39;could not fork&#39;); 
  } else if (!$pid) {
   //这里是子进程
  echo &#39;-&#39;;
  exit();
  }else{
  //这里是父进程
 echo &#39;A&#39;;
  pcntl_wait($status);//父进程执行到这里等等子进程执行完再执行
 echo &#39;B&#39;;
  }
 
运行结果:
 
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)

的作用。

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

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn