45fan.com - 路饭网

搜索: 您的位置主页 > 电脑频道 > 编程代码 > 阅读资讯:如何通过php实现多线程?

如何通过php实现多线程?

2016-03-05 17:50:02 来源:www.45fan.com 【

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


本文地址:http://www.45fan.com/bcdm/41383.html
Tags: 实现 PHP 三种
编辑:路饭网
推广内容
推荐阅读
热门推荐
推荐文章
关于我们 | 联系我们 | 友情链接 | 网站地图 | Sitemap | App | 返回顶部