>  기사  >  백엔드 개발  >  PHP 다중 스레드 시뮬레이션 구현의 세 가지 방법 소개

PHP 다중 스레드 시뮬레이션 구현의 세 가지 방법 소개

黄舟
黄舟원래의
2017-11-11 13:40:452419검색

프로젝트를 진행하면서 몇 가지 요구 사항이 있었는데, 특히 데이터 응답 처리에 많은 시간이 걸렸습니다. PHP 자체가 멀티스레딩을 지원하지 않는다는 사실을 모두 알고 있는데, PHP에서 멀티스레딩을 어떻게 구현해야 할까요? ?

1. Linux에서의 PHP 멀티스레딩

다음은 PHP의 pcntl_fork 함수에서 파생된 것입니다. 이 함수는 운영 체제 포크의 구현에 따라 달라지므로 이 문서에서 언급한 사항만 적용됩니다. 리눅스/unix로. 먼저 이 함수의 사용법을 살펴보겠습니다.

<?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를 통해 하위 프로세스를 생성하면 하위 프로세스 생성이 실패한 것입니다. 성공하면 프로세스 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에 Go를 추가하면 됩니다.

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\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&#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!\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){//创建的子进程 
  
      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);

요약: PHP 멀티스레드에서는 사용할 수 없습니다. 사실 우리 모두가 알고 있지만 다양한 방법을 통해 구현할 수 있습니다. 이 기사에서는 PHP의 멀티스레딩 구현 방법을 소개합니다. 이것이 도움이 되기를 바랍니다!

관련 권장 사항:

php 멀티 스레딩 구현 예


php 멀티 스레딩 구현 방법 - shell


PHP 멀티 스레딩 소형 케이스


PHP 멀티 스레딩 처리 문제 정보

위 내용은 PHP 다중 스레드 시뮬레이션 구현의 세 가지 방법 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.