通俗易懂的php多線程解決方案

2021-11-20 11:36 更新

我們在做項目的時候,有些需求,特別是數(shù)據(jù)的響應(yīng)處理需要花費大量的時間,由于php是一個短生命周期的腳本語言,到了默認的30秒,php的數(shù)據(jù)處理還沒完成,php的生命周期就結(jié)束了。這時需要使用異步并發(fā)處理策略,也就是說,一次php調(diào)用可以發(fā)出的多個請求,這些請求不是按照順序執(zhí)行,而是可以異步并發(fā)執(zhí)行的,一些請求用于在后臺處理數(shù)據(jù),一些請求用于接受后臺響應(yīng)狀態(tài),根據(jù)狀態(tài),與用戶做一些簡單的交互。但是問題來了,我們都知道php本身是不支持多線程的,那么應(yīng)該怎么實現(xiàn)php的多線程呢?

一、php模擬實現(xiàn)多線程的三種方法

1、linux下的php多線程

下面所講的東西是源自php的pcntl_fork函數(shù).因為這個函數(shù)依賴操作系統(tǒng)fork的實現(xiàn),所以本文所講的東西只適用于linux/unix。那么先看看這個函數(shù)的用法吧.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創(chuàng)建一個子進程,如果返回值是-1的話,那么說明子進程創(chuàng)建失敗.創(chuàng)建成功的進程id會返回給父進程,0返回給子進程.不好理解吧,所以應(yīng)該這樣寫:

<?php

$pid = pcntl_fork();

if($pid == -1){

         //創(chuàng)建失敗咱就退出唄,沒啥好說的

         die('could not fork');

}

else{

        if($pid){

                //從這里開始寫的代碼是父進程的,因為寫的是系統(tǒng)程序,記得退出的時候給個返回值

                exit(0);

        }

        else{

                //從這里開始寫的代碼都是在新的進程里執(zhí)行的,同樣正常退出的話,最好也給一個返回值

                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\r\nHost:" . $host . "\r\n\r\n";  

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

        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){//創(chuàng)建的子進程 

  

      static $num=0; 

      $num++; 

      if($num>=$number){ 

        //當(dāng)進程數(shù)量達到一定數(shù)量時候,就對子進程進行回收。 

        pcntl_wait($status); 

  

        $num--; 

      }  

    }else{ //為0 則代表是子進程創(chuàng)建的,則直接進入工作狀態(tài) 

  

      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); 

二、真正實現(xiàn)php多線程的方法

php真正的多線程實現(xiàn)方式,通過安裝php的擴展 pthread 可以做到。
但是這個下載的是 版本3 也就是php 7 才能用的,我們需要使的是 版本2
20151210175359990(1)

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

20151210175422414(2)

20151210175502827(3)

下一頁找到版本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


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號