php中curl新手秒變高手的使用教程實(shí)例
起先cURL是做為一種命令行工具設(shè)計(jì)出來(lái)的,比較幸運(yùn)的是,php也支持cURL了。通過(guò)cURL這個(gè)利器,我們能在php程序中自由地發(fā)送HTTP請(qǐng)求到某個(gè)url來(lái)獲取或者提交數(shù)據(jù),并且支持其它多種協(xié)議,比如FTP,Telnet以及SMTP等。在這篇博文中,我將簡(jiǎn)述下,在php中具體怎么使用cURL來(lái)處理一些事情。
一、curl的優(yōu)勢(shì)
你也許會(huì)說(shuō),在php中可以很容易的獲取某個(gè)url的內(nèi)容,只要通過(guò)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");
沒(méi)錯(cuò),以上函數(shù)在某些情況下使用起來(lái)確實(shí)很方便,但是我感覺(jué)這幾個(gè)函數(shù)不夠靈活,也沒(méi)法進(jìn)行錯(cuò)誤處理。而且,如果遇到要在php程序中向某個(gè)服務(wù)器提交表單數(shù)據(jù),上傳文件,處理cookies或者認(rèn)證等任務(wù)時(shí),以上三個(gè)函數(shù)根本無(wú)法勝任。這個(gè)時(shí)候,cURL就體現(xiàn)它的價(jià)值了。
cURl不但支持很多的網(wǎng)絡(luò)協(xié)議,而且提供了關(guān)于url請(qǐng)求的具體信息,很強(qiáng)大!
二、curl的簡(jiǎn)單使用步驟
要使用cURL來(lái)發(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è)步驟中。你可以通過(guò)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(shuō)過(guò),在向某個(gè)url發(fā)送get請(qǐng)求的話,沒(méi)有必要使用cURL來(lái)發(fā)送get請(qǐng)求,可以使用比較便捷的file_get_contents函數(shù)來(lái)完成請(qǐng)求。但是,一般地,我們?cè)谔峤荒硞€(gè)表單的時(shí)候,數(shù)據(jù)是通過(guò)post請(qǐng)求的內(nèi)容區(qū)域來(lái)提交的,而不是通過(guò)url參數(shù)來(lái)傳遞的, 這種情況下,我們應(yīng)該使用靈活的cURL來(lái)模擬發(fā)送post請(qǐng)求。
現(xiàn)在,讓我們使用cURL來(lái)模擬發(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中后,通過(guò)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)的來(lái)自post.php的回復(fù),最后輸出回復(fù)。上例雖然簡(jiǎn)單,但是充分演示了cURL發(fā)送post請(qǐng)求的便捷及強(qiáng)大之處,你可以在curl_setopt上做文章。
六、文件上傳
下面來(lái)看下如果通過(guò)cURL發(fā)送post請(qǐng)求來(lái)實(shí)現(xiàn)文件上傳。就拿深入淺出PHP下的文件上傳中的文件上傳例子來(lái)演示,在深入淺出php下的文件上傳中,是通過(guò)表單的提交來(lái)實(shí)現(xiàn)文件上傳的,那么通過(guò)cURL怎么來(lái)實(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);
通過(guò)以上示例代碼,可以將我本地機(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
)
)
由此可見(jiàn),如果你要通過(guò)cURL來(lái)上傳文件的話,只需要將上傳的文件路徑作為post數(shù)據(jù)設(shè)置到curl請(qǐng)求中,并且在路徑前面加上@符合。
七、文件下載
上述將了文件上傳,同樣的也可以使用curl來(lái)自動(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文件,我們需要將其下載下來(lái),并且保存到本地文件中,就可以嘗試使用下面的代碼來(lái)實(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)容寫(xiě)入本地文件 file_put_contents($path, $data);
注意:我以上省略了錯(cuò)誤處理方面的代碼,只是簡(jiǎn)單做個(gè)示例, 在實(shí)際中,你還需要通過(guò)curl_getinfo函數(shù)來(lái)進(jìn)行錯(cuò)誤處理!
上述代碼對(duì)于下載比較大型的文件是不適用的,因?yàn)樾枰葘⑽募x取到內(nèi)存中,等所有內(nèi)容都讀取完畢,然后再寫(xiě)入到本地硬盤(pán)中。即使php中設(shè)置的memory limit非常大,這種情況對(duì)性能的影響也是很大的。所以,我們對(duì)于大型文件的下載,應(yīng)該讓curl來(lái)接管這個(gè)任務(wù),實(shí)現(xiàn)邊下載,邊寫(xiě)入的處理,這樣的話,就沒(méi)什么問(wèn)題了。請(qǐng)看下述代碼:
$url = 'http://www.***.com/test.zip'; $path = 'local/path/to/test.zip'; // 打開(kāi)本地文件 $fp = fopen($path, 'w'); // 告訴curl本地文件句柄 $ch = curl_init($url); curl_setopt($ch, CURLOPT_FILE, $fp); curl_exec($ch); curl_close($ch); fclose($fp);
在上述代碼中,我們先打開(kāi)個(gè)本地文件,并將文件句柄設(shè)置到curl中,然后讓curl一邊讀取遠(yuǎn)程數(shù)據(jù),一邊寫(xiě)入到本地文件中。因?yàn)槲覀儾恍枰诔绦蛑蝎@取遠(yuǎn)程回復(fù)的內(nèi)容了,所以只要執(zhí)行請(qǐng)求就可以。
八、http 驗(yàn)證
如果服務(wù)器端需要驗(yàn)證請(qǐng)求,可以通過(guò)類似一下示例代碼來(lái)實(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);
九、通過(guò)代理發(fā)送請(qǐng)求
cURL還可以通過(guò)代理服務(wù)器來(lái)向發(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ù)
最后,我們來(lái)看下通過(guò)cURL來(lái)想服務(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);
// 通過(guò)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來(lái)將接受到的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)過(guò)json_decode以后,就轉(zhuǎn)換為php中的數(shù)據(jù)格式,成為了一個(gè)對(duì)象,所以可以通過(guò)$json_data->email來(lái)訪問(wèn)其中email字段的值,最后也就是輸出***@gmail.com。你可以使用上述代碼測(cè)試一下。
如果通過(guò)以下php數(shù)組生成json字符串的話:
$data = array('***', 'http://www.***.com', '***@gmail.com');
所生成的json字符串如下:
'["***","http:\/\/www.***.com","***@gmail.com"]'
上述json字符串在經(jīng)過(guò)json_decode處理后,就會(huì)變成php中的數(shù)組格式,如果要獲取email的話,就可以通過(guò)$json_data[2]來(lái)訪問(wèn)。
十一、cURL批處理(multi cURL)
cURL還有一個(gè)高級(jí)特性——批處理句柄(handle)。這一特性允許你同時(shí)或異步地打開(kāi)多個(gè)URL連接。
下面是來(lái)自來(lái)自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);
這里要做的就是打開(kāi)多個(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ù)是無(wú)隔斷(non-blocking)的,但會(huì)盡可能少地執(zhí)行。它返回一個(gè)狀態(tài)值,只要這個(gè)值等于常量 CURLM_CALL_MULTI_PERFORM ,就代表還有一些刻不容緩的工作要做(例如,把對(duì)應(yīng)URL的http頭信息發(fā)送出去)。也就是說(shuō),我們需要不斷調(diào)用該函數(shù),直到返回值發(fā)生改變。
而接下來(lái)的 while 循環(huán),只在 $active 變量為 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的用途,其中示例代碼是比較簡(jiǎn)單的。但是,相信你看完后應(yīng)該有使用cURL的沖動(dòng)了吧! 那就自己去找相關(guān)資料,手冊(cè)進(jìn)行測(cè)試吧!
好了,就寫(xiě)到這里吧!謝謝你的耐心閱讀!
附:
<?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)證證書(shū)來(lái)源的檢查
curl_setopt($ci, CURLOPT_SSL_VERIFYHOST, 2); // 從證書(shū)中檢查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-04
PHP通過(guò)bypass disable functions執(zhí)行系統(tǒng)命令的方法匯總
這篇文章主要介紹了PHP通過(guò)bypass disable functions執(zhí)行系統(tǒng)命令的方法匯總,需要的朋友可以參考下2018-05-05
Yii2.0實(shí)現(xiàn)生成二維碼功能實(shí)例
這篇文章主要介紹了Yii2.0實(shí)現(xiàn)生成二維碼功能,結(jié)合實(shí)例形式分析了Yii2.0框架生成二維碼所涉及的相關(guān)擴(kuò)展引入步驟與使用方法,需要的朋友可以參考下2017-10-10
thinkPHP+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-06
Laravel5.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-01
yii2組件之下拉框帶搜索功能的示例代碼(yii-select2)
本篇文章主要介紹了yii2組件之下拉框帶搜索功能的示例代碼(yii-select2),具有一定的參考價(jià)值,有興趣的可以了解一下2017-09-09
php實(shí)現(xiàn)學(xué)生管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了php學(xué)生管理系統(tǒng)的具體實(shí)現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07

