ホームページ  >  記事  >  バックエンド開発  >  シンプルなphpマルチスレッドソリューション

シンプルなphpマルチスレッドソリューション

烟雨青岚
烟雨青岚転載
2020-06-16 13:15:404811ブラウズ

シンプルなphpマルチスレッドソリューション

シンプルな PHP マルチスレッド ソリューション

プロジェクトを実行しているとき、いくつかのニーズ、特にデータが必要になります。 PHP はライフサイクルが短いスクリプト言語であるため、デフォルトの 30 秒を経過すると、PHP のデータ処理が完了せず、PHP のライフサイクルが終了します。

現時点では、非同期同時処理戦略、つまり 1 回の PHP 呼び出しで発行できる複数のリクエストを使用する必要があります。これらのリクエストは順番に実行されませんが、非同期に実行でき、一部のリクエストはバックグラウンドのプロセス データで使用され、一部のリクエストはバックグラウンドの応答ステータスを受け入れ、ステータスに基づいてユーザーとの単純な対話を行うために使用されます。

しかし、ここで問題が発生します。PHP 自体がマルチスレッドをサポートしていないことは誰もが知っていますが、PHP でマルチスレッドを実装するにはどうすればよいでしょうか?

1. マルチスレッドを実現するための PHP シミュレーションの 3 つの方法

1. Linux での PHP マルチスレッド

ソースは次のとおりです。 PHP の pcntl_fork 関数この関数はオペレーティング システムのフォークの実装に依存するため、この記事で説明する内容は Linux/Unix にのみ適用されます。 PHP マニュアルには次のように書かれています:

<?php
$pid = pcntl_fork();
if ($pid == -1) {
         die(&#39;could not fork&#39;);
} 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(&#39;could not fork&#39;);
}
else{
        if($pid){
                //从这里开始写的代码是父进程的,因为写的是系统程序,记得退出的时候给个返回值
                exit(0);
        }
        else{
                //从这里开始写的代码都是在新的进程里执行的,同样正常退出的话,最好也给一个返回值
                exit(0);
        }
}
?>

この変更は、親プロセスが子プロセスが正常に終了したことを知りたい場合は、前の pcntl_wait を追加すると理解しやすくなります。

2. stream_socket_client メソッド経由

function sendStream() { 
    $english_format_number = number_format($number, 4, &#39;.&#39;, &#39;&#39;); 
  
    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][&#39;body&#39;] = NoticeOrder::getSendData($v[&#39;waybill_id&#39;]); 
  
      } else { 
        $send_data[$k][&#39;body&#39;] = array($v[&#39;order_id&#39;] => array(&#39;extra&#39; => 16));  
      }  
      $data = json_encode($send_data[$k][&#39;body&#39;]); 
      $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&#39;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(&#39;h:i:s&#39;) . ".<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(&#39;worker&#39;,array(1),2);

2. php でマルチスレッドを実現する本当の方法

php でマルチスレッドを実現する本当の方法は、php 拡張機能 pthread をインストールすることで実現できます。

ここをクリックして https://github.com/krakjoe/pthreads をダウンロードします。ただし、このダウンロードはバージョン 3 であり、php 7 でのみ使用できます。バージョン 2 を使用する必要があります

シンプルなphpマルチスレッドソリューション

次に、次のようにページを更新し、一番下までドラッグします。

シンプルなphpマルチスレッドソリューション

シンプルなphpマルチスレッドソリューション

でバージョン 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

Note : コンパイル時に 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

Restart php /etc/init.d/php-fpm restart

読んでくれた皆さん、ありがとうございます。たくさんの利益が得られることを願っています。

推奨チュートリアル:「

php チュートリアル

以上がシンプルなphpマルチスレッドソリューションの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事は51dev.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。