当前位置:首页 > PHP教程 > php应用 > 列表

PHP之curl_multi并发详解方法

发布:smiling 来源: PHP粉丝网  添加日期:2022-06-20 08:16:51 浏览: 评论:0 

PHP中curl_multi并发详解

PHP中的curl_multi系列函数可以实现同时请求多个URL来实现并发,而不是像普通curl函数那样请求后会阻塞,直到结果返回才进行下一个请求。因此在批量请求URL时可通过curl_multi系列函数提升程序的运行效率。

curl普通请求

  1. $startTime = microtime(true); 
  2.  
  3. $chArr = []; 
  4.  
  5. $optArr = [ 
  6.  
  7.     CURLOPT_URL => 'http://www.httpbin.org/ip'
  8.  
  9.     CURLOPT_HEADER => 0, 
  10.  
  11.     CURLOPT_RETURNTRANSFER => 1, 
  12.  
  13. ]; 
  14.  
  15. $result = []; 
  16.  
  17. //创建多个curl资源并执行 
  18.  
  19. for ($i=0; $i<10; $i++) { 
  20.  
  21.     $chArr[$i] = curl_init(); 
  22.  
  23.     curl_setopt_array($chArr[$i], $optArr); 
  24.  
  25.     $result[$i] = curl_exec($chArr[$i]); 
  26.  
  27.     curl_close($chArr[$i]); 
  28.  
  29.  
  30. $endTime = microtime(true); 
  31.  
  32. echo sprintf("use time: %.3f s".PHP_EOL, $endTime - $startTime);use time: 6.080 s 
  33.  
  34. use time: 6.080 s 

curl_multi并发请求

  1. $startTime = microtime(true); 
  2.  
  3. $chArr = []; 
  4.  
  5. $optArr = [ 
  6.  
  7.     CURLOPT_URL => 'http://www.httpbin.org/ip'
  8.  
  9.     CURLOPT_HEADER => 0, 
  10.  
  11.     CURLOPT_RETURNTRANSFER => 1, 
  12.  
  13. ]; 
  14.  
  15. $result = []; 
  16.  
  17. //创建多个curl资源 
  18.  
  19. for ($i=0; $i<10; $i++) { 
  20.  
  21.     $chArr[$i] = curl_init(); 
  22.  
  23.     curl_setopt_array($chArr[$i], $optArr); 
  24.  
  25.  
  26. //创建批处理curl句柄 
  27.  
  28. $mh = curl_multi_init(); 
  29.  
  30. //将单个curl句柄添加到批处理curl句柄中 
  31.  
  32. foreach ($chArr as $ch) { 
  33.  
  34.     curl_multi_add_handle($mh$ch); 
  35.  
  36.  
  37. //判断操作是否仍在执行的标识的引用 
  38.  
  39. $active = null; 
  40.  
  41. /** 
  42.  
  43.  * 本次循环第一次处理 $mh 批处理中的 $ch 句柄,并将 $mh 批处理的执行状态写入 $active, 
  44.  
  45.  * 当状态值等于 CURLM_CALL_MULTI_PERFORM 时,表明数据还在写入或读取中,执行循环, 
  46.  
  47.  * 当第一次 $ch 句柄的数据写入或读取成功后,状态值变为 CURLM_OK ,跳出本次循环,进入下面的大循环中。 
  48.  
  49.  */ 
  50.  
  51. do { 
  52.  
  53.     //处理在批处理栈中的每一个句柄 
  54.  
  55.     $mrc = curl_multi_exec($mh$active); 
  56.  
  57. while ($mrc == CURLM_CALL_MULTI_PERFORM); 
  58.  
  59. /** 
  60.  
  61.  * 上面这段代码中,是可以直接使用 $active > 0 来作为 while 的条件,如下: 
  62.  
  63.  * do { 
  64.  
  65.  *   $mrc = curl_multi_exec($mh, $active); 
  66.  
  67.  * } while ($active > 0); 
  68.  
  69.  * 此时如果整个批处理句柄没有全部执行完毕时,系统会不停的执行 curl_multi_exec 函数,从而导致系统CPU占用会很高, 
  70.  
  71.  * 因此一般不采用这种方案,可以通过 curl_multi_select 函数来达到没有需要读取的程序就阻塞住的目的。 
  72.  
  73.  */ 
  74.  
  75. /** 
  76.  
  77.  * $active 为 true 时,即 $mh 批处理之中还有 $ch 句柄等待处理, 
  78.  
  79.  * $mrc == CURLM_OK,即上一次 $ch 句柄的读取或写入已经执行完毕。 
  80.  
  81.  */ 
  82.  
  83. while ($active && $mrc == CURLM_OK) { 
  84.  
  85.     /**  
  86.  
  87.      * 程序进入阻塞状态,直到批处理中有活动连接(即 $mh 批处理中还有可执行的 $ch 句柄), 
  88.  
  89.      * 这样执行的好处是 $mh 批处理中的 $ch 句柄会在读取或写入数据结束后($mrc == CURLM_OK)进入阻塞阶段, 
  90.  
  91.      * 而不会在整个 $mh 批处理执行时不停地执行 curl_multi_exec 函数,白白浪费CPU资源。 
  92.  
  93.      */ 
  94.  
  95.      if (curl_multi_select($mh) != -1) { 
  96.  
  97.         //程序退出阻塞状态继续执行需要处理的 $ch 句柄 
  98.  
  99.         do { 
  100.  
  101.             $mrc = curl_multi_exec($mh$active); 
  102.  
  103.         } while ($mrc == CURLM_CALL_MULTI_PERFORM); 
  104.  
  105.     } 
  106.  
  107.  
  108. foreach ($chArr as $i=>$ch) { 
  109.  
  110.     //获取某个curl句柄的返回值 
  111.  
  112.     $result[$i] = curl_multi_getcontent($ch); 
  113.  
  114.     //移除批处理句柄中的某个句柄资源 
  115.  
  116.     curl_multi_remove_handle($mh$ch); 
  117.  
  118.  
  119. //关闭一组curl句柄 
  120.  
  121. curl_multi_close($mh); 
  122.  
  123. $endTime = microtime(true); 
  124.  
  125. echo sprintf("use time: %.3f s".PHP_EOL, $endTime - $startTime); 
  126.  
  127. use time: 0.599 s 

通过对比上述程序的运行时间可以得知,使用curl_multi系列函数并发请求要比普通的curl函数依次请求效率高很多。

Tags: curl_multi PHP并发

分享到: