zoukankan      html  css  js  c++  java
  • PHP中curl_multi并发详解【转】

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

    curl普通请求

    $startTime = microtime(true);
    $chArr = [];
    $optArr = [
        CURLOPT_URL => 'http://www.httpbin.org/ip',
        CURLOPT_HEADER => 0,
        CURLOPT_RETURNTRANSFER => 1,
    ];
    $result = [];
    
    //创建多个curl资源并执行
    for ($i=0; $i<10; $i++) {
        $chArr[$i] = curl_init();
        curl_setopt_array($chArr[$i], $optArr);
        $result[$i] = curl_exec($chArr[$i]);
        curl_close($chArr[$i]);
    }
    
    $endTime = microtime(true);
    echo sprintf("use time: %.3f s".PHP_EOL, $endTime - $startTime);
    
    

    use time: 6.080 s

    curl_multi并发请求

    $startTime = microtime(true);
    $chArr = [];
    $optArr = [
        CURLOPT_URL => 'http://www.httpbin.org/ip',
        CURLOPT_HEADER => 0,
        CURLOPT_RETURNTRANSFER => 1,
    ];
    $result = [];
    
    //创建多个curl资源
    for ($i=0; $i<10; $i++) {
        $chArr[$i] = curl_init();
        curl_setopt_array($chArr[$i], $optArr);
    }
    //创建批处理curl句柄
    $mh = curl_multi_init();
    //将单个curl句柄添加到批处理curl句柄中
    foreach ($chArr as $ch) {
        curl_multi_add_handle($mh, $ch);
    }
    //判断操作是否仍在执行的标识的引用
    $active = null;
    /**
     * 本次循环第一次处理 $mh 批处理中的 $ch 句柄,并将 $mh 批处理的执行状态写入 $active,
     * 当状态值等于 CURLM_CALL_MULTI_PERFORM 时,表明数据还在写入或读取中,执行循环,
     * 当第一次 $ch 句柄的数据写入或读取成功后,状态值变为 CURLM_OK ,跳出本次循环,进入下面的大循环中。
     */
    do {
        //处理在批处理栈中的每一个句柄
        $mrc = curl_multi_exec($mh, $active);
    } while ($mrc == CURLM_CALL_MULTI_PERFORM);
    /**
     * 上面这段代码中,是可以直接使用 $active > 0 来作为 while 的条件,如下:
     * do {
     *   $mrc = curl_multi_exec($mh, $active);
     * } while ($active > 0);
     * 此时如果整个批处理句柄没有全部执行完毕时,系统会不停的执行 curl_multi_exec 函数,从而导致系统CPU占用会很高,
     * 因此一般不采用这种方案,可以通过 curl_multi_select 函数来达到没有需要读取的程序就阻塞住的目的。
     */
    
    /**
     * $active 为 true 时,即 $mh 批处理之中还有 $ch 句柄等待处理,
     * $mrc == CURLM_OK,即上一次 $ch 句柄的读取或写入已经执行完毕。
     */
    while ($active && $mrc == CURLM_OK) {
        /** 
         * 程序进入阻塞状态,直到批处理中有活动连接(即 $mh 批处理中还有可执行的 $ch 句柄),
         * 这样执行的好处是 $mh 批处理中的 $ch 句柄会在读取或写入数据结束后($mrc == CURLM_OK)进入阻塞阶段,
         * 而不会在整个 $mh 批处理执行时不停地执行 curl_multi_exec 函数,白白浪费CPU资源。
         */
         if (curl_multi_select($mh) != -1) {
            //程序退出阻塞状态继续执行需要处理的 $ch 句柄
            do {
                $mrc = curl_multi_exec($mh, $active);
            } while ($mrc == CURLM_CALL_MULTI_PERFORM);
        }
    }
    
    foreach ($chArr as $i=>$ch) {
        //获取某个curl句柄的返回值
        $result[$i] = curl_multi_getcontent($ch);
        //移除批处理句柄中的某个句柄资源
        curl_multi_remove_handle($mh, $ch);
    }
    //关闭一组curl句柄
    curl_multi_close($mh);
    $endTime = microtime(true);
    echo sprintf("use time: %.3f s".PHP_EOL, $endTime - $startTime);
    
    

    use time: 0.599 s

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

    文章转自:https://juejin.im/post/5ed9f4c2e51d450d800a3479
     
    更多文章参考如下:
    http://www.php.cn/php-weizijiaocheng-409819.html   //php如何实现并发请求(代码)

    http://www.php.cn/php-weizijiaocheng-405038.html   //PHP使用curl_multi实现并发请求的方法示例php技巧

    https://www.php.net/manual/zh/function.curl-multi-init.php  //php curl_multi

    http://lampblog.org/category/phpdev  //使用PHP并发执行任务–curl_multi应用

    http://www.php.cn/php-weizijiaocheng-393305.html  //PHP中使用CURL之php curl详细解析和常见大坑

  • 相关阅读:
    2013/11/21工作随笔-PHP开启多进程
    php中mysql操作的buffer知识
    你不一定懂的cpu显示信息
    好文收藏系列(三)
    doctrine2到底是个什么玩意
    制作火焰图(纯笔记)
    《精通Linux内核必会的75个绝技》知识杂记
    BIG5, GB(GB2312, GBK, ...), Unicode编码, UTF8, WideChar, MultiByte, Char说明与区别
    sed替换换行符“ ”
    mysql将字符串字段转为数字排序或比大小
  • 原文地址:https://www.cnblogs.com/KillBugMe/p/13117880.html
Copyright © 2011-2022 走看看