目錄
PHP模拟发送POST请求之五curl基本使用和多线程优化,之五curl
首頁 後端開發 php教程 PHP模拟发送POST请求之五curl基本使用和多线程优化,之五curl_PHP教程

PHP模拟发送POST请求之五curl基本使用和多线程优化,之五curl_PHP教程

Jul 12, 2016 am 09:05 AM
curl

PHP模拟发送POST请求之五curl基本使用和多线程优化,之五curl

今天来介绍PHP模拟发送POST请求的重型武器——cURL函数库的使用和其多线程的优化方法。

说起cURL函数,可谓是老生常谈,但网上许多资料都在关键部分语焉不详,列出一大堆手册上的东西,搞得我入门时也万分痛苦,我翻了些资料,结合自己的笔记,总结了这篇博文,希望能给初次接触cURL的开发者提供一些帮助。


 

cURL的基本使用步骤

首先来介绍cURL:

cURL按照HTTP头信息来模拟浏览器传输数据,它支持FTP, FTPS, HTTP, HTTPS, DICT, FILE等协议,拥有HTTPS认证,HTTP POST方法,HTTP PUT方法,FTP上传,HTTP上传,代理服务器,cookies,用户名/密码认证等功能。cURL可谓是实现爬站抓取网页、POST数据等功能的利器。

使用cURL函数主要分为四部分:

1.初始化cURL。

2.设置cURL变量,这是cRUL的核心,扩展功能全靠这一步骤。

3.执行cURL,获取结果。

4.关闭连接,回收资源。

<span>$ch</span> = curl_init();<span>//</span><span>1</span>
<span>
curl_setopt(</span><span>$ch</span>, CURLOPT_URL, "http://localhost");<span>//</span><span>2</span>

<span>$output</span> = curl_exec(<span>$ch</span>);<span>//</span><span>3</span>
<span>
curl_close(</span><span>$ch</span>);<span>//</span><span>4</span>
登入後複製

另外,我们还可以使用curl_getinfo($ch)函数获取curl执行的信息,其结果为一个数组

$info数组的内容包括如下等内容:

  • “url” //资源网络地址
  • “content_type” //内容编码
  • “http_code” //HTTP状态码
  • “filetime” //文件创建时间
  • “total_time” //总耗时
  • “size_upload” //上传数据的大小
  • “size_download” //下载数据的大小
  • “speed_download” //下载速度
  • “speed_upload” //上传速度
  • “download_content_length”//下载内容的长度
  • “upload_content_length” //上传内容的长度

cURL的常用设置

下面详细介绍curl使用第二步时常用的变量设置,在使用curl函数时,可以按各种需求来设置。

设置基本信息:

curl_setopt($ch, CURLOPT_URL, $string);//设置curl的目录地址

curl_setopt($ch, CURLOPT_PORT, $port);//设置连接端口,一般不设置默认80

curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);//返回结果流,并不是输出它留待后续处理,一般会设置这一项,稍后处理抓取到的信息,而不是直接输出出来。

设置POST数据信息:

curl_setopt($ch, CURLOPT_POST, 1);//设置传送数据方式为POST

curl_setopt($ch, CURLOPT_POSTFIELDS, $string);//设置要传输的数据

设置验证信息:

curl_setopt($ch, CURLOPT_COOKIE, $string);//设置curl执行时所携带的Cookie信息

curl_setopt($ch, CURLOPT_USERAGENT, $string);//设置curl模拟的浏览器信息

curl_setopt($ch, CURLOPT_REFERER, $string);//设置header中的referer,有利于破解防盗链

curl_setopt($ch, CURLOPT_USERPWD, $string);//传递一个连接中需要的用户名和密码,格式为:"[username]:[password]"

curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);//设置允许服务器重定向

设置强化信息:

curl_setopt($ch, CURLOPT_NOBODY, 1);//设置不允许输出HTML body体,如果在抓取页面标题等信息时设置此选项会大大加快速度

curl_setopt($ch, CURLOPT_TIMEOUT, $int);//设置允许执行的最长秒数(超时时间),设置数值小的时候,CURL会放弃执行时间长的页面

curl_setopt($ch, CURLOPT_HEADER, 1);//设置允许将读取目标时产生的header头文件包含在输出流中


cURL批处理功能的基本使用

当然cURL的功能并不止于此,在手册上你可以找到它更多的变量设置。而且cURL最强大的地方在其批处理功能。

cURL的批处理似乎也很好理解,以下是一般步骤:

1.$mh = curl_multi_init();//初始化一个批处理句柄。

2.curl_multi_add_handle($mh,$ch); //往批处理句柄中添加设置好的$ch句柄。

3.curl_multi_exec($mh,$running);//执行$mh句柄,并将$mh句柄的运行状态写入$running变量中

4.当$running为true时循环执行curl_multi_close()函数

5.循环结束后遍历$mh句柄,用curl_multi_getcontent()获取第一个句柄的返回值

6.用curl_multi_remove_handle()将$mh中的句柄移除

7.用curl_multi_close()关闭$mh批处理句柄。

代码如下:

 <?<span>php 

        </span><span>$chArr</span>=<span>[];

        </span><span>for</span>(<span>$i</span>=0;<span>$i</span><50;<span>$i</span>++<span>){

              </span><span>$chArr</span>[<span>$i</span>]=curl_init("http://www.baidu.com"<span>);

              curl_setopt(</span><span>$chArr</span>[<span>$i</span>],CURLOPT_RETURNTRANSFER,1<span>);

        }

       </span><span>$mh</span> = curl_multi_init(); <span>//</span><span>1</span>

       <span>foreach</span>(<span>$chArr</span> <span>as</span> <span>$k</span> => <span>$ch</span><span>){      

         curl_multi_add_handle(</span><span>$mh</span>,<span>$ch</span>); <span>//</span><span>2</span>
<span>    
    }

       </span><span>$running</span> = <span>null</span><span>; 

       </span><span>do</span><span>{ 

           curl_multi_exec(</span><span>$mh</span>,<span>$running</span>); <span>//</span><span>3</span>
<span>
        }</span><span>while</span>(<span>$running</span> > 0); <span>//</span><span>4</span>

        <span>foreach</span>(<span>$chArr</span> <span>as</span> <span>$k</span> => <span>$ch</span><span>){ 

              </span><span>$result</span>[<span>$k</span>]= curl_multi_getcontent(<span>$ch</span>); <span>//</span><span>5</span>
<span>
              curl_multi_remove_handle(</span><span>$mh</span>,<span>$ch</span>);<span>//</span><span>6</span>
<span>
        }

        curl_multi_close(</span><span>$mh</span>); <span>//</span><span>7</span>

      ?>    
登入後複製


cURL批处理时内存占用过多的问题

但是,执行大批量的句柄时我们会发现一个很严重的问题,那就是执行时系统CPU占用率几乎100%,几乎是死机状态了。纠其原因,那是因为在$running>0,执行 curl_multi_exec($mh,$running)而整个批处理句柄没有全部执行完毕时,系统会不停地执行curl_multi_exec()函数。我们用实验来证明:

我们在循环中curl_multi_exec($mh,$running)句前加入一个echo "a";的语句。我们的目的是执行50次对百度的访问,然后来看一下结果。

从图中滚动条的大小(滚动条已经最小状态了)可以大概看出输出a的个数,500个也不止,所以我们便可以找到占用CPU的罪魁祸首了。


cURL批处理时的内存优化方案

进行改动的方式是应用curl函数库中的curl_multi_select()函数,其函数原型如下:

<p>int curl_multi_select ( resource $mh [, float $timeout = 1.0 ] )</p>
<p>阻塞直到cURL批处理连接中有活动连接。成功时返回描述符集合中描述符的数量。失败时,select失败时返回-1,否则返回超时(从底层的select系统调用)。</p>
登入後複製

我用们curl_multi_select()函数来达到没有需要读取的程序就阻塞住的目的。

我们对批处理的第3、4步进行优化,利用其多线程,模拟并发程序。

很多朋友会对手册中提供的代码心存疑惑(我一开始也是),下面的代码及解释。

<span>$running</span> = <span>null</span><span>;

</span><span>do</span><span> {

    </span><span>$mrc</span> = curl_multi_exec(<span>$mh</span>, <span>$running</span><span>);

} </span><span>while</span> (<span>$mrc</span> ==<span> CURLM_CALL_MULTI_PERFORM);

</span><span>//</span><span>本次循环第一次处理$mh批处理中的$ch句柄,并将$mh批处理的执行状态写入$running,当状态值等于CURLM_CALL_MULTI_PERFORM时,表明数据还在写入或读取中,执行循环,当第一次$ch句柄的数据写入或读取成功后,状态值变为CURLM_OK,跳出本次循环,进入下面的大循环之中。

//$running为true,即$mh批处理之中还有$ch句柄正待处理,$mrc==CURLM_OK,即上一次$ch句柄的读取或写入已经执行完毕。</span>

<span>while</span> (<span>$running</span> && <span>$mrc</span> ==<span> CURLM_OK) { 

    </span><span>if</span> (curl_multi_select(<span>$mh</span>) != -1) {<span>//</span><span>$mh批处理中还有可执行的$ch句柄,curl_multi_select($mh) != -1程序退出阻塞状态。</span>

        <span>do</span> {                            <span>//</span><span>继续执行需要处理的$ch句柄。</span>

             <span>$mrc</span> = curl_multi_exec(<span>$mh</span>, <span>$running</span><span>);

        } </span><span>while</span> (<span>$mrc</span> ==<span> CURLM_CALL_MULTI_PERFORM);

    }

}</span>
登入後複製

这样执行的好处是$mh批处理中的$ch句柄会在读取或写入数据结束后($mrc==CURLM_OK),进入curl_multi_select($mh)的阻塞阶段,而不会在整个$mh批处理执行时不停地执行curl_multi_exec,白白浪费CPU资源。


cURL批处理的内存优化结果

完整代码如下:

<?<span>php 

        </span><span>$chArr</span>=<span>[];

        </span><span>for</span>(<span>$i</span>=0;<span>$i</span><50;<span>$i</span>++<span>){

        </span><span>$chArr</span>[<span>$i</span>]=curl_init("http://www.baidu.com"<span>);

        curl_setopt(</span><span>$chArr</span>[<span>$i</span>],CURLOPT_RETURNTRANSFER,1<span>);

        }

       </span><span>$mh</span> =<span> curl_multi_init(); 

        </span><span>foreach</span>(<span>$chArr</span> <span>as</span> <span>$k</span> => <span>$ch</span><span>)      

        curl_multi_add_handle(</span><span>$mh</span>,<span>$ch</span><span>); <br />
        </span><span>$running</span> = <span>null</span><span>; 

        </span><span>do</span><span> {
<br /></span><span>        $mrc</span> = curl_multi_exec(<span>$mh</span>, <span>$running</span><span>);

     } </span><span>while</span> (<span>$mrc</span> ==<span> CURLM_CALL_MULTI_PERFORM);


    </span><span>while</span> (<span>$running</span> && <span>$mrc</span> ==<span> CURLM_OK) {

         </span><span>if</span> (curl_multi_select(<span>$mh</span>) != -1<span>) {

             </span><span>do</span><span> {

                  </span><span>$mrc</span> = curl_multi_exec(<span>$mh</span>, <span>$running</span><span>);

             } </span><span>while</span> (<span>$mrc</span> ==<span> CURLM_CALL_MULTI_PERFORM);

         }

    }

       </span><span>foreach</span>(<span>$chArr</span> <span>as</span> <span>$k</span> => <span>$ch</span><span>){ 

               </span><span>$result</span>[<span>$k</span>]= curl_multi_getcontent(<span>$ch</span><span>); 

               curl_multi_remove_handle(</span><span>$mh</span>,<span>$ch</span><span>);

       }

        curl_multi_close(</span><span>$mh</span><span>); 

      </span>?>
登入後複製

我们再次在 $mrc = curl_multi_exec($mh, $running)句子前加入echo "a";结果如下图:

虽然也不止50次,但是比之未优化前,CPU使用率已经大为改观。

虽然curl函数非常强大,但是我们还是有使用其他函数来发送POST请求的机会,另外也能从更底层了解curl函数,所以本辑也用大很大篇幅在其他函数上。

OK,本辑结束,写这辑博文的同时,我也学习到了很多。如果您觉得本博文对您有帮助,请您点推荐或关注我,我们继续分享我的笔记总结。如果有什么问题,您可以在下方留言讨论,谢谢阅读。

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/1067637.htmlTechArticlePHP模拟发送POST请求之五curl基本使用和多线程优化,之五curl 今天来介绍PHP模拟发送POST请求的重型武器cURL函数库的使用和其多线程的优化方...
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

python中CURL和python requests的相互轉換如何實現 python中CURL和python requests的相互轉換如何實現 May 03, 2023 pm 12:49 PM

curl和Pythonrequests都是發送HTTP請求的強大工具。雖然curl是一種命令列工具,可讓您直接從終端機發送請求,但Python的請求庫提供了一種更具程式化的方式來從Python程式碼發送請求。將curl轉換為Pythonrequestscurl指令的基本語法如下所示:curl[OPTIONS]URL將curl指令轉換為Python請求時,我們需要將選項和URL轉換為Python程式碼。這是一個範例curlPOST指令:curl-XPOSThttps://example.com/api

PHP8.1發布:引入curl多個請求並發處理 PHP8.1發布:引入curl多個請求並發處理 Jul 08, 2023 pm 09:13 PM

PHP8.1發布:引入curl多個請求並發處理近日,PHP官方發布了最新版本的PHP8.1,其中引入了一個重要的特性:curl多個請求並發處理。這個新功能為開發者提供了一個更有效率和靈活的方式來處理多個HTTP請求,大大提升了效能和使用者體驗。在以往的版本中,處理多個請求往往需要透過建立多個curl資源,並使用循環來分別發送和接收資料。這種方式雖然能夠實現目

從頭到尾:如何使用php擴充cURL進行HTTP請求 從頭到尾:如何使用php擴充cURL進行HTTP請求 Jul 29, 2023 pm 05:07 PM

從頭到尾:如何使用php擴充cURL進行HTTP請求引言:在Web開發中,經常需要與第三方API或其他遠端伺服器進行通訊。而使用cURL進行HTTP請求是一種常見且強大的方式。本文將介紹如何使用php擴充cURL來執行HTTP請求,並提供一些實用的程式碼範例。一、準備工作首先,請確保php已安裝cURL擴充。可以在命令列執行php-m|grepcurl查

Linux下更新curl版本教程! Linux下更新curl版本教程! Mar 07, 2024 am 08:30 AM

在Linux下更新curl版本,您可以按照以下步驟進行操作:檢查目前curl版本:首先,您需要確定目前系統中安裝的curl版本。開啟終端,並執行以下指令:curl--version該指令將顯示目前curl的版本資訊。確認可用的curl版本:在更新curl之前,您需要確定可用的最新版本。您可以造訪curl的官方網站(curl.haxx.se)或相關的軟體來源,尋找最新版本的curl。下載curl原始碼:使用curl或瀏覽器,下載您選擇的curl版本的原始碼檔案(通常為.tar.gz或.tar.bz2

PHP Curl中如何處理網頁的 301 重定向? PHP Curl中如何處理網頁的 301 重定向? Mar 08, 2024 am 11:36 AM

PHPCurl中如何處理網頁的301重定向?使用PHPCurl發送網路請求時,常會遇到網頁回傳的301狀態碼,表示頁面被永久重定向。為了正確處理這種情況,我們需要在Curl請求中加入一些特定的選項和處理邏輯。以下將詳細介紹在PHPCurl中如何處理網頁的301重定向,並提供具體的程式碼範例。 301重定向處理原理301重定向是指伺服器回傳了一個30

linux curl是什麼 linux curl是什麼 Apr 20, 2023 pm 05:05 PM

在linux中,curl是一個非常實用的、用來與伺服器之間傳輸資料的工具,是一個利用URL規則在命令列下工作的檔案傳輸工具;它支援檔案的上傳和下載,是綜合傳輸工具。 curl提供了一大堆非常有用的功能,包括代理存取、使用者認證、ftp上傳下載、HTTP POST、SSL連線、cookie支援、斷點續傳等等。

php curl怎麼設定cookie php curl怎麼設定cookie Sep 26, 2021 am 09:27 AM

php curl設定cookie的方法:1、建立PHP範例檔;2、透過「curl_setopt」函數設定cURL傳輸選項;3、在CURL中傳遞cookie即可。

深入了解PHP Curl中的301跳轉機制 深入了解PHP Curl中的301跳轉機制 Mar 08, 2024 pm 01:21 PM

PHP中的Curl是一個強大的工具,用於與不同的伺服器進行通訊。在實際應用中,經常會遇到301跳轉的情況,即伺服器會重定向請求。本文將深入探討PHPCurl中的301跳轉機制,並提供具體的程式碼範例,幫助讀者更好地理解並應用該功能。什麼是301跳轉301跳轉是伺服器端發出的一種重定向指令,意味著所要求的資源已經被永久移動到另一個位置。當瀏覽器或客戶端發送請求

See all articles