PHP中利用pcntl进行多进程并发控制

 

PHP中利用pcntl进行多进程并发控制

分类: PHP   7921人阅读  评论(1)  收藏  举报

pcntl_fork可以很方便的创建进程,对于一般的需要固定的多进程处理的应用场景来说,实现比较简单,但是,对于需要大量并发创建子进程的应用场景来说,主要的问题在于会产生大量的僵尸进程。。。

 

我们的应用中,之前是采用将过程中产生的子进程pid收集起来, 间隔一定时间统一回收(pcntl_waitpid),这样带来的一个问题是:在大量并发情况下,服务器压力过大,会导致子进程“死掉”,这个时候,子进程不是僵尸,无法回收掉,主控进程就卡在那里不动了。。。

 

今天借鉴了一下“http://www.perkiset.org/forum/php/forked_php_daemon_example-t474.0.html”文中的方式,用二次创建进程的方式(孙进程处理业务)来将子进程交给系统,达到了在父进程中不需要wait真正的业务子进程的目的,也更接近其他语言中的线程并发控制:

 

我们原来的子进程创建接口:

[php]  view plain copy
  1. /** 
  2.  * author: selfimpr 
  3.  * blog: http://blog.youkuaiyun.com/lgg201 
  4.  * mail: lgg860911@yahoo.com.cn 
  5.  */  
  6. function new_child($func_name) {  
  7.     $args = func_get_args();  
  8.     unset($args[0]);  
  9.     $pid = pcntl_fork();  
  10.     if($pid == 0) {  
  11.         function_exists($func_nameand exit(call_user_func_array($func_name$args)) or exit(-1);  
  12.     } else if($pid == -1) {  
  13.         echo "Couldn’t create child process.";  
  14.     } else {  
  15.         return $pid;  
  16.     }  
  17. }  

 

新的方式:

[php]  view plain copy
  1. <?php  
  2. /** 
  3.  * author: selfimpr 
  4.  * blog: http://blog.youkuaiyun.com/lgg201 
  5.  * mail: lgg860911@yahoo.com.cn 
  6.  */  
  7. function daemon($func_name) {  
  8.     $args = func_get_args();  
  9.     unset($args[0]);  
  10.     $pid = pcntl_fork();  
  11.     if($pid) {  
  12.         pcntl_wait($status);  
  13.     } else if($pid == -1) {  
  14.         echo "Couldn't create child process.";  
  15.     } else {  
  16.         $pid = pcntl_fork();  
  17.         if($pid == 0) {  
  18.             posix_setsid();  
  19.             function_exists($func_nameand exit(call_user_func_array($func_name$args)) or exit(-1);  
  20.         } else if($pid == -1) {  
  21.             echo "Couldn’t create child process.";  
  22.         } else {  
  23.             exit;  
  24.         }  
  25.     }  
  26. }  
  27. ?>  

 

新方式的完整测试:

测试方法:cli模式下运行, ps -ef | grep php查看过程中的php进程情况

[php]  view plain copy
  1. <?php  
  2. /** 
  3.  * author: selfimpr 
  4.  * blog: http://blog.youkuaiyun.com/lgg201 
  5.  * mail: lgg860911@yahoo.com.cn 
  6.  */  
  7. function daemon($func_name) {  
  8.     $args = func_get_args();  
  9.     unset($args[0]);  
  10.     $pid = pcntl_fork();  
  11.     if($pid) {  
  12.         pcntl_wait($status);  
  13.     } else if($pid == -1) {  
  14.         echo "Couldn't create child process.";  
  15.     } else {  
  16.         $pid = pcntl_fork();  
  17.         if($pid == 0) {  
  18.             posix_setsid();  
  19.             function_exists($func_nameand exit(call_user_func_array($func_name$args)) or exit(-1);  
  20.         } else if($pid == -1) {  
  21.             echo "Couldn’t create child process.";  
  22.         } else {  
  23.             sleep(5);  
  24.             exit;  
  25.         }  
  26.     }  
  27. }  
  28. function test() {  
  29.     while($i ++ < 10) {  
  30.         echo "child process $i/n";  
  31.         sleep(1);  
  32.     }  
  33. }  
  34. daemon(test);  
  35. while(++ $jecho "parent process $j/n";  
  36. ?>  

php ”Call to undefined function pcntl_fork”错误 解决办法  

2010-08-09 15:05:39|  分类: 实习工作|举报|字号 订阅

下载LOFTER客户端

 执行下面的函数时,php报了”Call to undefined function pcntl_fork”错误:

PHP代码
    
<?php      
$pid = pcntl_fork();      
       
if ($pid == -1) {      
     die("could not fork");      
} else if ($pid) {      
     // we are the parent      
} else {      
     // we are the child      
}      
?>; 


原来是因为php默认没有安装pcntl扩展包,你可以缷载掉原来的,并用./configure –enable-pcntl重新编译php安装。

当然,你也可以不缷载php,而是直接编译pcntl扩展,并在php.ini文件中加载它即可:

代码
    
# 下载php源码并解压      
cd php-5.3.2/ext/pcntl      
phpize5      
./configure --with-php-config=/usr/bin/php-config5      
make && make install 


安装完成后会输出*.so文件被编译在什么目录:Installing shared extensions:  /usr/lib/php5/20060613+lfs/。/usr/lib/php5/20060613+lfs是php默认的扩展程序目录,所以你不需要在php.ini文件中额外的定义php扩展目录。

加载pcntl扩展:

代码
    
# 新建文件pcntl.ini      
sudo vim /etc/php5/conf.d/pcntl.ini      
# pcntl.ini的内容      
extension=pcntl.so 

最后重启php-cgi进程与Nginx:

代码
    
sudo /etc/init.d/php stop      
sudo /etc/init.d/php start      
sudo /etc/init.d/nginx restart 


在命令行输入:

代码
    
php -r 'var_dump(function_exists("gzopen"));' 


如果输出bool(true),则表示pcntl扩展已经加载成功了。当然你也可以创建一个php文件,然后通过访问它来测试pcntl是否安装成功,文件内容如下:

PHP代码
    
<?php      
  echo function_exists("pcntl_fork");      
?>  文章来自: 王殿臣-天空网(bestlovesky.com)详文参考:http://www.bestlovesky.com/read.php/950.htm


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值