找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 1361|回复: 0
打印 上一主题 下一主题

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

[复制链接]

2588

主题

2588

帖子

7694

积分

论坛元老

Rank: 8Rank: 8

积分
7694
跳转到指定楼层
楼主
发表于 2018-2-14 05:41:55 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

            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') . ".
  ";
            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实现多线程类似的方法,希望对大家的学习有所帮助。
            
            
您可能感兴趣的文章:
  • php异步多线程swoole用法实例
  • PHP多线程类及用法实例
  • PHP多线程编程之管道通信实例分析
  • 如何解决PHP无法实现多线程的问题
  • php多线程实现方法及用法实例详解
  • PHP使用Pthread实现的多线程操作实例
  • PHP安装threads多线程扩展基础教程
  • PHP7多线程搭建教程
            
  • 分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    用户反馈
    客户端