简单的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.0Host:" . $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教程》

更多相关文章

  1. 分享几个 Hyperf 常用助手函数
  2. 关于PHP进程防止内存溢出的排查
  3. PHP 数组常用函数总结
  4. PHP中箭头函数的实例详解
  5. 收好100个最常用的PHP函数
  6. 基于ob系列函数实现PHP网站页面静态化
  7. PHP 进程管理器 PHP-FPM
  8. 直击PHP array_reverse() 函数原理及实例解析
  9. 看懂PHP进程管理器php-fpm

随机推荐

  1. Socket Android手机客户端与PC服务端局域
  2. sscanf函数引起android 5.0卡死,C++中慎用
  3. Android 系统基础
  4. Android之Window类简介
  5. 强烈推荐Android新手的入门学习策略之一,A
  6. Android下添加新的自定义键值和按键处理
  7. Android输入框限制字符长度,字母自动大写
  8. Android体系框架
  9. Android KitKat 4.4 Wifi移植AP模式和网
  10. [Android]判断是否是华为EMUI系统