php三种实现多线程类似的方法

5年以前  |  阅读数:641 次  |  编程语言:PHP 

1、curl_multi方法

当需要多线程的时候,可以用curl_multi一次性请求多个操作来完成,但curl走的是网络通信,效率与可靠性就比较差了的。


    function main(){ 

       $sql = "select waybill_id,order_id from waybill where status>40 order by update_time desc limit 10 "; 

        $data = Yii::app()->db->createCommand($sql)->queryAll(); //yii 框架格式 

        foreach ($data as $k => $v) {  

          if ($k % 2 == 0) { //偶数发一个网址 

            $send_data[$k]['url'] = ''; 

            $send_data[$k]['body'] = $v['waybill_id']; 

          } else { //奇数发送另外一个网址 
            $send_data[$k]['url'] = 'http://www.abc.com'; 

            $send_data[$k]['body']=array($v['order_id'] => array('extra' => 16)); 

          } 

        } 

        $back_data =sendMulitRequest($send_data); 

        var_dump($back_data); 

      } 
      function sendMulitRequest($send_data){ 
        $params = array(); 
        $curl = $text = array(); 
        $handle = curl_multi_init(); 

        foreach ($data as $k => $v) { 

          if (empty($v['url'])) { 

            $v['url'] = "http://www.xxx.com"; //if url is empty,set defalut url 

          } 

          $reqBody = json_encode($v['body']); 

          $reqStream = array( 

            'body' => $reqBody, 
          );  
          $encRequest = base64_encode(json_encode($reqStream));  
          $params['data'] = $encRequest; 
          $curl[$k] = curl_init(); 
          curl_setopt($curl[$k], CURLOPT_URL, $v['url']); 
          curl_setopt($curl[$k], CURLOPT_POST, TRUE); 
          curl_setopt($curl[$k], CURLOPT_HEADER, 0); 
          curl_setopt($curl[$k], CURLOPT_POSTFIELDS, http_build_query($params)); 
          curl_setopt($curl[$k], CURLOPT_RETURNTRANSFER, 1); 
          curl_multi_add_handle($handle, $curl[$k]); 
        } 
        $active = null; 

        do { 

          $mrc = curl_multi_exec($handle, $active); 

        } while ($mrc == CURLM_CALL_MULTI_PERFORM); 
        while ($active && $mrc == CURLM_OK) { 

          if (curl_multi_select($handle) != -1) { 

            do { 

              $mrc = curl_multi_exec($handle, $active); 

            } while ($mrc == CURLM_CALL_MULTI_PERFORM); 

          } 

        } 
        foreach ($curl as $k => $v) {  
          if (curl_error($curl[$k]) == "") { 
            $text[$k] = (string) curl_multi_getcontent($curl[$k]); 
           }  
          curl_multi_remove_handle($handle, $curl[$k]);  
          curl_close($curl[$k]); 
        } 
        curl_multi_close($handle);  
        return $text;  
      } 

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){//创建的子进程 

          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实现多线程类似的方法,希望对大家的学习有所帮助。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8