首页 行业资讯 宠物日常 宠物养护 宠物健康 宠物故事

有什么办法可以实现php控制多线程运行

发布网友 发布时间:2022-04-22 13:00

我来回答

3个回答

懂视网 时间:2022-04-06 14:38

简单的php多线程解决方法

我们在做项目的时候,有些需求,特别是数据的响应处理需要花费大量的时间,由于php是一个短生命周期的脚本语言,到了默认的30秒,php的数据处理还没完成,php的生命周期就结束了。

这时需要使用异步并发处理策略,也就是说,一次php调用可以发出的多个请求,这些请求不是按照顺序执行,而是可以异步并发执行的,一些请求用于在后台处理数据,一些请求用于接受后台响应状态,根据状态,与用户做一些简单的交互。

但是问题来了,我们都知道php本身是不支持多线程的,那么应该怎么实现php的多线程呢?

一、php模拟实现多线程的三种方法

1、linux下的php多线程

下面所讲的东西是源自php的pcntl_fork函数.因为这个函数依赖操作系统fork的实现,所以本文所讲的东西只适用于linux/unix。那么先看看这个函数的用法吧.php手册上是这么说的:

<?php
$pid = pcntl_fork();
if ($pid == -1) {
  die('could not fork');
} else if ($pid) {
  // we are the parent
  pcntl_wait($status); //Protect against Zombie children
} else {
  // we are the child
}
?>

通过pcntl_fork创建一个子进程,如果返回值是-1的话,那么说明子进程创建失败.创建成功的进程id会返回给父进程,0返回给子进程.不好理解吧,所以应该这样写:

<?php
$pid = pcntl_fork();
if($pid == -1){
  //创建失败咱就退出呗,没啥好说的
  die('could not fork');
}
else{
 if($pid){
  //从这里开始写的代码是父进程的,因为写的是系统程序,记得退出的时候给个返回值
  exit(0);
 }
 else{
  //从这里开始写的代码都是在新的进程里执行的,同样正常退出的话,最好也给一个返回值
  exit(0);
 }
}
?>

这样一改好理解多了,如果你父进程希望知道子进程正常退出的话,可以加上前面的pcntl_wait。

2.通过stream_socket_client 方式

function sendStream() { 
 $english_format_number = number_format($number, 4, '.', ''); 
 
 echo $english_format_number; 
 exit(); 
 $timeout = 10; 
 $result = array(); 
 $sockets = array(); 
 $convenient_read_block = 8192; 
 $host = "test.local.com"; 
 $sql = "select waybill_id,order_id from xm_waybill where status>40 order by update_time desc limit 1 "; 
 $data = Yii::app()->db->createCommand($sql)->queryAll(); 
 $id = 0; 
 
 foreach ($data as $k => $v) { 
 if ($k % 2 == 0) { 
 $send_data[$k]['body'] = NoticeOrder::getSendData($v['waybill_id']); 
 
 } else { 
 $send_data[$k]['body'] = array($v['order_id'] => array('extra' => 16)); 
 } 
 $data = json_encode($send_data[$k]['body']); 
 $s = stream_socket_client($host . ":80", $errno, $errstr, $timeout, STREAM_CLIENT_ASYNC_CONNECT | STREAM_CLIENT_CONNECT); 
 if ($s) { 
 $sockets[$id++] = $s; 
 $http_message = "GET /php/test.php?data=" . $data . " HTTP/1.0
Host:" . $host . "
"; 
 fwrite($s, $http_message); 
 } else { 
 echo "Stream " . $id . " failed to open correctly."; 
 } 
 } 
 
 while (count($sockets)) { 
 
 $read = $sockets; 
 
 stream_select($read, $w = null, $e = null, $timeout); 
 if (count($read)) { 
 /* stream_select generally shuffles $read, so we need to 
  compute from which socket(s) we're reading. */
 foreach ($read as $r) { 
 
  $id = array_search($r, $sockets); 
  $data = fread($r, $convenient_read_block); 
  if (strlen($data) == 0) { 
  echo "Stream " . $id . " closes at " . date('h:i:s') . ".<br> "; 
  fclose($r); 
  unset($sockets[$id]); 
  } else { 
  $result[$id] = $data; 
  } 
 } 
 } else { 
 /* A time-out means that *all* streams have failed 
  to receive a response. */
 echo "Time-out!
"; 
 break; 
 } 
 } 
 print_r($result); 
 
 }

3、通过多进程代替多线程

function daemon($func_name,$args,$number){ 
 while(true){ 
 $pid=pcntl_fork(); 
 if($pid==-1){ 
 echo "fork process fail"; 
 exit(); 
 }elseif($pid){//创建的子进程 
 
 static $num=0; 
 $num++; 
 if($num>=$number){ 
 //当进程数量达到一定数量时候,就对子进程进行回收。 
 pcntl_wait($status); 
 
 $num--; 
 } 
 }else{ //为0 则代表是子进程创建的,则直接进入工作状态 
 
 if(function_exists($func_name)){ 
 while (true) { 
  $ppid=posix_getpid(); 
  var_dump($ppid); 
  call_user_func_array($func_name,$args); 
  sleep(2); 
 } 
 }else{ 
 echo "function is not exists"; 
 } 
 exit(); 
 } 
 } 
} 
function worker($args){ 
 //do something 
 
} 
daemon('worker',array(1),2);

二、真正实现php多线程的方法

php真正的多线程实现方式,通过安装php的扩展 pthread 可以做到。

点此下载https://github.com/krakjoe/pthreads 但是这个下载的是 版本3 也就是php 7 才能用的,我们需要使的是 版本2

6092113c1728601303604028ac954b6.png

然后刷新的页面如下,拖到最底部:

4e53194eaffe80a031c2341653b4f32.png

5d15360dd1731d6e963293d76d50ce9.png

下一页找到版本2的

下载下来,这个v2 才是php5才可以使用的

下载下来,安装:

或者,您直接这样下载:

cd /tools 
 wget https://github.com/krakjoe/pthreads/archive/v2.0.10.zip 
 unzip v2.0.10.zip 
 cd pthreads-2.0.10 
 /usr/local/php/bin/phpize 
 ./configure --with-php-config=/usr/local/php/bin/php-config 
 make 
 make install

注意:您的php 在编译的时候需要开启 –enable-maintainer-zts

./configure --prefix=/usr/local/php --disable-fileinfo --enable-fpm --with-config-file-path=/etc --with-config-file-scan-dir=/etc/php.d --with-openssl --with-zlib --with-curl --enable-ftp --with-gd --with-xmlrpc --with-jpeg-dir --with-png-dir --with-freetype-dir --enable-gd-native-ttf --enable-mbstring --with-mcrypt=/usr/local/libmcrypt --enable-zip --with-mysql=/usr/local/mysql --without-pear --enable-maintainer-zts

vim /etc/php.ini 
添加
extension=pthreads.so

重启php /etc/init.d/php-fpm restart

感谢大家的阅读,希望大家收益多多。

推荐教程:《php教程》

热心网友 时间:2022-04-06 11:46

  php(做为现在的主流开发语言)中实现多线程? 看到这个标题, 你一定以为我疯了..但是事实上我真的这么做了.
  下面是我的一些做法, 已经实验过. 确实可以的.
  我们知道php(做为现在的主流开发语言)本身是不支持多线程的, 但是我们的WEB服务器是支持多线程的.
  也就是说可以同时让多人一起访问. 这也是我在php(做为现在的主流开发语言)中实现多线程的基础.
  假设我们现在运行的是a.php(做为现在的主流开发语言)这个文件. 但是我在程序中又请求WEB服务器运行另一个b.php(做为现在的主流开发语言)
  那么这两个文件将是同时执行的.
  (PS: 一个链接请求发送之后, WEB服务器就会执行它, 而不管客户端是否已经退出)
  有些时候, 我们想运行的不是另一个文件, 而是本文件中的一部分代码.该怎么办呢?
  其实可是通过参数来控制a.php(做为现在的主流开发语言)来运行哪一段程序.
  下面看一个例子:
  //a.php(做为现在的主流开发语言)

php(做为现在的主流开发语言)代码:--------------------------------------------------------------------------------
  <?php(做为现在的主流开发语言)
function runThread()
{
$fp = fsockopen(localhost, 80, $errno, $errmsg);

fputs($fp, "GET /a.php(做为现在的主流开发语言)?act=b "); //这里的第二个参数是HTTP协议中规定的请求头
//不明白的请看RFC中的定义

fclose($fp);
}
  function a()
{
$fp = fopen(result_a.log, w);
fputs($fp, Set in . Date(h:i:s, time()) . (double)microtime() . " ");

fclose($fp);
}
  function b()
{
$fp = fopen(result_b.log, w);
fputs($fp, Set in . Date(h:i:s, time()) . (double)microtime() . " ");

热心网友 时间:2022-04-06 13:04

没有办法,PHP没有多线程,这是底层的病

声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com