php中curl新手秒變高手的使用教程實(shí)例
起先cURL是做為一種命令行工具設(shè)計(jì)出來的,比較幸運(yùn)的是,php也支持cURL了。通過cURL這個(gè)利器,我們能在php程序中自由地發(fā)送HTTP請(qǐng)求到某個(gè)url來獲取或者提交數(shù)據(jù),并且支持其它多種協(xié)議,比如FTP,Telnet以及SMTP等。在這篇博文中,我將簡述下,在php中具體怎么使用cURL來處理一些事情。
一、curl的優(yōu)勢
你也許會(huì)說,在php中可以很容易的獲取某個(gè)url的內(nèi)容,只要通過file_get_contents,file或者readfile函數(shù)就能輕松實(shí)現(xiàn),根本不必使用cURL:
$content = file_get_contents("http://www.***.com"); $lines = file("http://www.***.com"); readfile("http://www.***.com");
沒錯(cuò),以上函數(shù)在某些情況下使用起來確實(shí)很方便,但是我感覺這幾個(gè)函數(shù)不夠靈活,也沒法進(jìn)行錯(cuò)誤處理。而且,如果遇到要在php程序中向某個(gè)服務(wù)器提交表單數(shù)據(jù),上傳文件,處理cookies或者認(rèn)證等任務(wù)時(shí),以上三個(gè)函數(shù)根本無法勝任。這個(gè)時(shí)候,cURL就體現(xiàn)它的價(jià)值了。
cURl不但支持很多的網(wǎng)絡(luò)協(xié)議,而且提供了關(guān)于url請(qǐng)求的具體信息,很強(qiáng)大!
二、curl的簡單使用步驟
要使用cURL來發(fā)送url請(qǐng)求,具體步驟大體分為以下四步:
1.初始化
2.設(shè)置請(qǐng)求選項(xiàng)
3.執(zhí)行一個(gè)cURL會(huì)話并且獲取相關(guān)回復(fù)
4.釋放cURL句柄,關(guān)閉一個(gè)cURL會(huì)話
// 1. 初始化一個(gè)cURL會(huì)話 $ch = curl_init(); // 2. 設(shè)置請(qǐng)求選項(xiàng), 包括具體的url curl_setopt($ch, CURLOPT_URL, "http://www.***.com"); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_HEADER, 0); // 3. 執(zhí)行一個(gè)cURL會(huì)話并且獲取相關(guān)回復(fù) $response = curl_exec($ch); // 4. 釋放cURL句柄,關(guān)閉一個(gè)cURL會(huì)話 curl_close($ch);
cURL之所以強(qiáng)大,正是體現(xiàn)在第二個(gè)步驟中。你可以通過curl_setopt靈活地設(shè)置請(qǐng)求選項(xiàng),這里面有很多的可選項(xiàng)
三、錯(cuò)誤處理
在上述代碼中,你也可以增加錯(cuò)誤處理的代碼:
$response = curl_exec($ch); if ($response === FALSE) { echo "cURL 具體出錯(cuò)信息: " . curl_error($ch); }
注意了,在做上述判斷時(shí)務(wù)必要使用===
,因?yàn)檎?qǐng)求的回復(fù)可能是空字符串,curl在請(qǐng)求出錯(cuò)的情況下回返回FALSE值,所以我們必須使用===
,而不是==
。
四、獲取curl請(qǐng)求的具體信息
在執(zhí)行一個(gè)cURL請(qǐng)求后,你也可以使用curl_getinfo獲取該請(qǐng)求的具體信息:
curl_exec($ch); $curl_info= curl_getinfo($ch); echo "收到的http回復(fù)的code為: {$curl_info['http_code']}";
上述$curl_info是一個(gè)關(guān)聯(lián)數(shù)組,可以從中獲取很多的具體請(qǐng)求信息。參考http://cn2.php.net/manual/zh/function.curl-getinfo.php
五、使用curl發(fā)送post請(qǐng)求
我們?cè)谇懊嬲f過,在向某個(gè)url發(fā)送get請(qǐng)求的話,沒有必要使用cURL來發(fā)送get請(qǐng)求,可以使用比較便捷的file_get_contents
函數(shù)來完成請(qǐng)求。但是,一般地,我們?cè)谔峤荒硞€(gè)表單的時(shí)候,數(shù)據(jù)是通過post請(qǐng)求的內(nèi)容區(qū)域來提交的,而不是通過url參數(shù)來傳遞的, 這種情況下,我們應(yīng)該使用靈活的cURL來模擬發(fā)送post請(qǐng)求。
現(xiàn)在,讓我們使用cURL來模擬發(fā)送一個(gè)post請(qǐng)求到post.php腳本,提交幾個(gè)數(shù)據(jù)到post.php,然后在post.php中輸出post請(qǐng)求中的數(shù)據(jù)。示例代碼如下:
$url = "http://www.***.me/post.php"; $post_data = array ( "blog_name" => "***", "blog_url" => "http://www.***.com", "action" => "Submit" ); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設(shè)置請(qǐng)求為post類型 curl_setopt($ch, CURLOPT_POST, 1); // 添加post數(shù)據(jù)到請(qǐng)求中 curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data); // 執(zhí)行post請(qǐng)求,獲得回復(fù) $response= curl_exec($ch); curl_close($ch); echo $response;
以上請(qǐng)求發(fā)送到post.php中后,通過print_r($_POST)輸出后,以上示例代碼會(huì)輸出如下回復(fù):
Array ( [blog_name] => *** [blog_url] => http://www.***.com [action] => Submit )
正如我們看到的,cURL成功發(fā)送post請(qǐng)求到post.php,提交了一些數(shù)據(jù),并且收到了相應(yīng)的來自post.php的回復(fù),最后輸出回復(fù)。上例雖然簡單,但是充分演示了cURL發(fā)送post請(qǐng)求的便捷及強(qiáng)大之處,你可以在curl_setopt上做文章。
六、文件上傳
下面來看下如果通過cURL發(fā)送post請(qǐng)求來實(shí)現(xiàn)文件上傳。就拿深入淺出PHP下的文件上傳中的文件上傳例子來演示,在深入淺出php下的文件上傳中,是通過表單的提交來實(shí)現(xiàn)文件上傳的,那么通過cURL怎么來實(shí)現(xiàn)呢?
$url = "http://www.***.me/upload.php"; $post_data = array ( "attachment" => "@E:/jackblog/boy.jpg" ); //初始化cURL會(huì)話 $ch = curl_init(); //設(shè)置請(qǐng)求的url curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); //設(shè)置為post請(qǐng)求類型 curl_setopt($ch, CURLOPT_POST, 1); //設(shè)置具體的post數(shù)據(jù) curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data); $response = curl_exec($ch); curl_close($ch); print_r($response);
通過以上示例代碼,可以將我本地機(jī)器上的boy.jpg上傳到本地服務(wù)器的upload.php中,如果在upload.php輸出上傳的具體信息的話,以上示例代碼最后的輸出的回復(fù)為:
Array ( [attachment] => Array ( [name] => boy.jpg [type] => application/octet-stream [tmp_name] => D:\xampp\tmp\phpF27D.tmp [error] => 0 [size] => 11490 ) )
由此可見,如果你要通過cURL來上傳文件的話,只需要將上傳的文件路徑作為post數(shù)據(jù)設(shè)置到curl請(qǐng)求中,并且在路徑前面加上@符合。
七、文件下載
上述將了文件上傳,同樣的也可以使用curl來自動(dòng)地完成文件的下載以及保存。有一點(diǎn)要補(bǔ)充下,在執(zhí)行一個(gè)curl請(qǐng)求時(shí),如果你需要獲取返回的內(nèi)容,而不是直接輸出返回的內(nèi)容的話,別忘記使用下面的代碼設(shè)置,因?yàn)閏url的默認(rèn)是輸出請(qǐng)求的回復(fù)內(nèi)容:
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
假如在***的服務(wù)器根目錄下面有一個(gè)test.zip文件,我們需要將其下載下來,并且保存到本地文件中,就可以嘗試使用下面的代碼來實(shí)現(xiàn):
//設(shè)置請(qǐng)求的下載文件的url $url = 'http://www.***.com/test.zip'; //保存到本地的文件路徑 $path = 'local/path/to/test.zip'; //初始化請(qǐng)求,設(shè)置請(qǐng)求,獲取回復(fù),關(guān)閉會(huì)話 $ch = curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $data = curl_exec($ch); curl_close($ch); //將文件內(nèi)容寫入本地文件 file_put_contents($path, $data);
注意:我以上省略了錯(cuò)誤處理方面的代碼,只是簡單做個(gè)示例, 在實(shí)際中,你還需要通過curl_getinfo
函數(shù)來進(jìn)行錯(cuò)誤處理!
上述代碼對(duì)于下載比較大型的文件是不適用的,因?yàn)樾枰葘⑽募x取到內(nèi)存中,等所有內(nèi)容都讀取完畢,然后再寫入到本地硬盤中。即使php中設(shè)置的memory limit非常大,這種情況對(duì)性能的影響也是很大的。所以,我們對(duì)于大型文件的下載,應(yīng)該讓curl來接管這個(gè)任務(wù),實(shí)現(xiàn)邊下載,邊寫入的處理,這樣的話,就沒什么問題了。請(qǐng)看下述代碼:
$url = 'http://www.***.com/test.zip'; $path = 'local/path/to/test.zip'; // 打開本地文件 $fp = fopen($path, 'w'); // 告訴curl本地文件句柄 $ch = curl_init($url); curl_setopt($ch, CURLOPT_FILE, $fp); curl_exec($ch); curl_close($ch); fclose($fp);
在上述代碼中,我們先打開個(gè)本地文件,并將文件句柄設(shè)置到curl中,然后讓curl一邊讀取遠(yuǎn)程數(shù)據(jù),一邊寫入到本地文件中。因?yàn)槲覀儾恍枰诔绦蛑蝎@取遠(yuǎn)程回復(fù)的內(nèi)容了,所以只要執(zhí)行請(qǐng)求就可以。
八、http 驗(yàn)證
如果服務(wù)器端需要驗(yàn)證請(qǐng)求,可以通過類似一下示例代碼來實(shí)現(xiàn):
$url = "http://www.***.com/users/"; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設(shè)置用戶名以及密碼 curl_setopt($ch, CURLOPT_USERPWD, "username:password"); // 設(shè)置重導(dǎo)向 curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1); curl_setopt($ch, CURLOPT_UNRESTRICTED_AUTH, 1); $response = curl_exec($ch); curl_close($ch);
九、通過代理發(fā)送請(qǐng)求
cURL還可以通過代理服務(wù)器來向發(fā)送請(qǐng)求,請(qǐng)看一下示例代碼:
$ch = curl_init(); curl_setopt($ch, CURLOPT_URL,'http://www.***.com'); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設(shè)置代理ip地址 curl_setopt($ch, CURLOPT_PROXY, '222.73.173.50:8080'); // 要驗(yàn)證的話,這里設(shè)置用戶名以及密碼 curl_setopt($ch, CURLOPT_PROXYUSERPWD,'username:password'); $response = curl_exec($ch); curl_close ($ch);
十、發(fā)送json數(shù)據(jù)
最后,我們來看下通過cURL來想服務(wù)器端發(fā)送json數(shù)據(jù)。具體的代碼如下:
$url = 'http://www.***.me/json.php'; // 建立json字符串 $data = array('site' => '***', 'url' => 'http://www.***.com','email'=>'***@gmail.com'); $json_string = json_encode($data); $ch=curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 通過post請(qǐng)求發(fā)送上述json字符串 curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "POST"); curl_setopt($ch, CURLOPT_POSTFIELDS, array('data'=>$json_string)); $response = curl_exec($ch); curl_close($ch); echo $response;
大家可以看到,上述請(qǐng)求是發(fā)送到我的本地服務(wù)器的json.php下,我在該文件中使用json_decode來將接受到的json字符串轉(zhuǎn)換為對(duì)象,然后輸出其中的email字段,代碼如下:
$json_data = json_decode($_POST['data']); echo $json_data->email;
在上述代碼中接受的json字符串為:
'{"site":"***","url":"http:\/\/www.***.com","email":"***@gmail.com"}'
經(jīng)過json_decode以后,就轉(zhuǎn)換為php中的數(shù)據(jù)格式,成為了一個(gè)對(duì)象,所以可以通過$json_data->email
來訪問其中email字段的值,最后也就是輸出***@gmail.com。你可以使用上述代碼測試一下。
如果通過以下php數(shù)組生成json字符串的話:
$data = array('***', 'http://www.***.com', '***@gmail.com');
所生成的json字符串如下:
'["***","http:\/\/www.***.com","***@gmail.com"]'
上述json字符串在經(jīng)過json_decode處理后,就會(huì)變成php中的數(shù)組格式,如果要獲取email的話,就可以通過$json_data[2]
來訪問。
十一、cURL批處理(multi cURL)
cURL還有一個(gè)高級(jí)特性——批處理句柄(handle)。這一特性允許你同時(shí)或異步地打開多個(gè)URL連接。
下面是來自來自php.net的示例代碼:
// 創(chuàng)建兩個(gè)cURL資源 $ch1 = curl_init(); $ch2 = curl_init(); // 指定URL和適當(dāng)?shù)膮?shù) curl_setopt($ch1, CURLOPT_URL, "http://lxr.php.net/"); curl_setopt($ch1, CURLOPT_HEADER, 0); curl_setopt($ch2, CURLOPT_URL, "http://www.php.net/"); curl_setopt($ch2, CURLOPT_HEADER, 0); // 創(chuàng)建cURL批處理句柄 $mh = curl_multi_init(); // 加上前面兩個(gè)資源句柄 curl_multi_add_handle($mh,$ch1); curl_multi_add_handle($mh,$ch2); // 預(yù)定義一個(gè)狀態(tài)變量 $active = null; // 執(zhí)行批處理 do { $mrc = curl_multi_exec($mh, $active); } while ($mrc == CURLM_CALL_MULTI_PERFORM); while ($active && $mrc == CURLM_OK) { if (curl_multi_select($mh) != -1) { do { $mrc = curl_multi_exec($mh, $active); } while ($mrc == CURLM_CALL_MULTI_PERFORM); } } // 關(guān)閉各個(gè)句柄 curl_multi_remove_handle($mh, $ch1); curl_multi_remove_handle($mh, $ch2); curl_multi_close($mh);
這里要做的就是打開多個(gè)cURL句柄并指派給一個(gè)批處理句柄。然后你就只需在一個(gè)while循環(huán)里等它執(zhí)行完畢。
這個(gè)示例中有兩個(gè)主要循環(huán)。第一個(gè) do-while 循環(huán)重復(fù)調(diào)用 curl_multi_exec()
。這個(gè)函數(shù)是無隔斷(non-blocking)的,但會(huì)盡可能少地執(zhí)行。它返回一個(gè)狀態(tài)值,只要這個(gè)值等于常量 CURLM_CALL_MULTI_PERFORM
,就代表還有一些刻不容緩的工作要做(例如,把對(duì)應(yīng)URL的http頭信息發(fā)送出去)。也就是說,我們需要不斷調(diào)用該函數(shù),直到返回值發(fā)生改變。
而接下來的 while 循環(huán),只在 $activ
e 變量為 true 時(shí)繼續(xù)。這一變量之前作為第二個(gè)參數(shù)傳給了 curl_multi_exec()
,代表只要批處理句柄中是否還有活動(dòng)連接。接著,我們調(diào)用 curl_multi_select()
,在活動(dòng)連接(例如接受服務(wù)器響應(yīng))出現(xiàn)之前,它都是被“屏蔽”的。這個(gè)函數(shù)成功執(zhí)行后,我們又會(huì)進(jìn)入另一個(gè) do-while 循環(huán),繼續(xù)下一條URL。
十二、總結(jié)
在這篇博文中只是列舉了一些cURL的用途,其中示例代碼是比較簡單的。但是,相信你看完后應(yīng)該有使用cURL的沖動(dòng)了吧! 那就自己去找相關(guān)資料,手冊(cè)進(jìn)行測試吧!
好了,就寫到這里吧!謝謝你的耐心閱讀!
附:
<?php /** * @require curl-extension */ class SimpleHttpClient { private static $boundary = ''; public static function get($url, $params) { $url = $url . '?' . http_build_query($params); return self::http($url, 'GET'); } public static function post($url, $params, $files = array()) { $headers = array(); if (!$files) { $body = http_build_query($params); } else { $body = self::build_http_query_multi($params, $files); $headers[] = "Content-Type: multipart/form-data; boundary=" . self::$boundary; } return self::http($url, 'POST', $body, $headers); } /** * Make an HTTP request * * @return string API results * @ignore */ private static function http($url, $method, $postfields = NULL, $headers = array()) { try{ $ssl = stripos($url,'https://') === 0 ? true : false; $ci = curl_init(); /* Curl settings */ curl_setopt($ci, CURLOPT_USERAGENT, $_SERVER['HTTP_USER_AGENT']); //在HTTP請(qǐng)求中包含一個(gè)"User-Agent: "頭的字符串。 curl_setopt($ci, CURLOPT_CONNECTTIMEOUT, 30); curl_setopt($ci, CURLOPT_TIMEOUT, 30); curl_setopt($ci, CURLOPT_RETURNTRANSFER, TRUE); curl_setopt($ci, CURLOPT_ENCODING, ""); if ($ssl) { curl_setopt($ci, CURLOPT_SSL_VERIFYPEER, 0); // 對(duì)認(rèn)證證書來源的檢查 curl_setopt($ci, CURLOPT_SSL_VERIFYHOST, 2); // 從證書中檢查SSL加密算法是否存在 } curl_setopt($ci, CURLOPT_HEADER, FALSE); switch ($method) { case 'POST': curl_setopt($ci, CURLOPT_POST, TRUE); if (!empty($postfields)) { curl_setopt($ci, CURLOPT_POSTFIELDS, $postfields); } break; } curl_setopt($ci, CURLOPT_URL, $url ); curl_setopt($ci, CURLOPT_HTTPHEADER, $headers ); curl_setopt($ci, CURLINFO_HEADER_OUT, TRUE ); $response = curl_exec($ci); $httpCode = curl_getinfo($ci, CURLINFO_HTTP_CODE); $httpInfo = curl_getinfo($ci); if (FALSE === $response) throw new Exception(curl_error($ci), curl_errno($ci)); } catch(Exception $e) { throw $e; } //echo '<pre>'; //var_dump($response); //var_dump($httpInfo); curl_close ($ci); return $response; } private static function build_http_query_multi($params, $files) { if (!$params) return ''; $pairs = array(); self::$boundary = $boundary = uniqid('------------------'); $MPboundary = '--'.$boundary; $endMPboundary = $MPboundary. '--'; $multipartbody = ''; foreach ($params as $key => $value) { $multipartbody .= $MPboundary . "\r\n"; $multipartbody .= 'content-disposition: form-data; name="' . $key . "\"\r\n\r\n"; $multipartbody .= $value."\r\n"; } foreach ($files as $key => $value) { if (!$value) {continue;} if (is_array($value)) { $url = $value['url']; if (isset($value['name'])) { $filename = $value['name']; } else { $parts = explode( '?', basename($value['url'])); $filename = $parts[0]; } $field = isset($value['field']) ? $value['field'] : $key; } else { $url = $value; $parts = explode( '?', basename($url)); $filename = $parts[0]; $field = $key; } $content = file_get_contents($url); $multipartbody .= $MPboundary . "\r\n"; $multipartbody .= 'Content-Disposition: form-data; name="' . $field . '"; filename="' . $filename . '"'. "\r\n"; $multipartbody .= "Content-Type: image/unknown\r\n\r\n"; $multipartbody .= $content. "\r\n"; } $multipartbody .= $endMPboundary; return $multipartbody; } }
到此這篇關(guān)于php中curl新手秒變高手的使用教程實(shí)例的文章就介紹到這了,更多相關(guān)php中curl使用實(shí)例內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
ThinkPHP框架實(shí)現(xiàn)FTP圖片上傳功能示例
這篇文章主要介紹了ThinkPHP框架實(shí)現(xiàn)FTP圖片上傳功能,結(jié)合實(shí)例形式分析了thinkPHP框架實(shí)現(xiàn)ftp圖片上傳功能相關(guān)的配置、文件傳輸功能等操作技巧,需要的朋友可以參考下2019-04-04PHP通過bypass disable functions執(zhí)行系統(tǒng)命令的方法匯總
這篇文章主要介紹了PHP通過bypass disable functions執(zhí)行系統(tǒng)命令的方法匯總,需要的朋友可以參考下2018-05-05Yii2.0實(shí)現(xiàn)生成二維碼功能實(shí)例
這篇文章主要介紹了Yii2.0實(shí)現(xiàn)生成二維碼功能,結(jié)合實(shí)例形式分析了Yii2.0框架生成二維碼所涉及的相關(guān)擴(kuò)展引入步驟與使用方法,需要的朋友可以參考下2017-10-10thinkPHP+phpexcel實(shí)現(xiàn)excel報(bào)表輸出功能示例
這篇文章主要介紹了thinkPHP+phpexcel實(shí)現(xiàn)excel報(bào)表輸出功能,結(jié)合具體實(shí)例形式分析了thinkPHP整合PHPExcel實(shí)現(xiàn)針對(duì)Excel文件相關(guān)操作技巧,需要的朋友可以參考下2017-06-06Laravel5.1 框架登錄和注冊(cè)實(shí)現(xiàn)方法詳解
這篇文章主要介紹了Laravel5.1 框架登錄和注冊(cè)實(shí)現(xiàn)方法,結(jié)合實(shí)例形式詳細(xì)分析了laravel5.1框架登錄與注冊(cè)相關(guān)配置、路由、實(shí)現(xiàn)方法與操作注意事項(xiàng),需要的朋友可以參考下2020-01-01yii2組件之下拉框帶搜索功能的示例代碼(yii-select2)
本篇文章主要介紹了yii2組件之下拉框帶搜索功能的示例代碼(yii-select2),具有一定的參考價(jià)值,有興趣的可以了解一下2017-09-09php實(shí)現(xiàn)學(xué)生管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了php學(xué)生管理系統(tǒng)的具體實(shí)現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07