摘要: [目錄] php中的curl使用入門教程和常見用法實例 1、curl的優點 2、curl的簡單使用步驟 3、錯誤處理 4、獲取curl請求的具體信息 5、使用curl發送post請求 6、文件上傳 7、文件下載 8、http 驗證 9、經過代理髮送請求 10、發送json數據 11、cURL批處理(...php
[目錄] php中的curl使用入門教程和常見用法實例 1、curl的優點 2、curl的簡單使用步驟 3、錯誤處理 4、獲取curl請求的具體信息 5、使用curl發送post請求 6、文件上傳 7、文件下載 8、http 驗證 9、經過代理髮送請求 10、發送json數據 11、cURL批處理(multi cURL) 12、總結
起先cURL是作爲一種命令行工具設計出來的,比較幸運的是,php也支持cURL了。經過cURL這個利器,咱們能在php程序中自由地發送 HTTP請求到某個url來獲取或者提交數據,而且支持其它多種協議,好比FTP,Telnet以及SMTP等。在這篇博文中,我將簡述下,在php中具 體怎麼使用cURL來處理一些事情。算法
你也許會說,在php中能夠很容易的獲取某個url的內容,只要經過file_get_contents,file或者readfile函數就能輕鬆實現,根本沒必要使用cURL:json
$content = file_get_contents("http://www.52fhy.com"); $lines = file("http://www.52fhy.com"); readfile("http://www.52fhy.com");
沒錯,以上函數在某些狀況下使用起來確實很方便,可是我感受這幾個函數不夠靈活,也無法進行錯誤處理。並且,若是遇到要在php程序中向某個服務器 提交表單數據,上傳文件,處理cookies或者認證等任務時,以上三個函數根本沒法勝任。這個時候,cURL就體現它的價值了。數組
cURl不但支持不少的網絡協議,並且提供了關於url請求的具體信息,很強大!服務器
要使用cURL來發送url請求,具體步驟大致分爲如下四步:cookie
1.初始化
2.設置請求選項
3.執行一個cURL會話而且獲取相關回復
4.釋放cURL句柄,關閉一個cURL會話網絡
// 1. 初始化一個cURL會話 $ch = curl_init(); // 2. 設置請求選項, 包括具體的url curl_setopt($ch, CURLOPT_URL, "http://www.52fhy.com"); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_HEADER, 0); // 3. 執行一個cURL會話而且獲取相關回復 $response = curl_exec($ch); // 4. 釋放cURL句柄,關閉一個cURL會話 curl_close($ch);
cURL之因此強大,正是體如今第二個步驟中。你能夠經過curl_setopt靈活地設置請求選項,這裏面有不少的可選項,具體能夠參考:http://cn2.php.net/manual/zh/function.curl-setopt.phpapp
在上述代碼中,你也能夠增長錯誤處理的代碼:curl
$response = curl_exec($ch); if ($response === FALSE) { echo "cURL 具體出錯信息: " . curl_error($ch); }
注意了,在作上述判斷時務必要使用===
,由於請求的回覆多是空字符串,curl在請求出錯的狀況下回返回FALSE值,因此咱們必須使用===
,而不是==
。異步
在執行一個cURL請求後,你也可使用curl_getinfo獲取該請求的具體信息:
curl_exec($ch); $curl_info= curl_getinfo($ch); echo "收到的http回覆的code爲: {$curl_info['http_code']}";
上述$curl_info是一個關聯數組,能夠從中獲取不少的具體請求信息。參考http://cn2.php.net/manual/zh/function.curl-getinfo.php
咱們在前面說過,在向某個url發送get請求的話,沒有必要使用cURL來發送get請求,可使用比較便捷的file_get_contents
函數來完成請求。可是,通常地,咱們在提交某個表單的時候,數據是經過post請求的內容區域來提交的,而不是經過url參數來傳遞的, 這種狀況下,咱們應該使用靈活的cURL來模擬發送post請求。
如今,讓咱們使用cURL來模擬發送一個post請求到post.php腳本,提交幾個數據到post.php,而後在post.php中輸出post請求中的數據。示例代碼以下:
$url = "http://www.52fhy.me/post.php"; $post_data = array ( "blog_name" => "52fhy", "blog_url" => "http://www.52fhy.com", "action" => "Submit" ); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設置請求爲post類型 curl_setopt($ch, CURLOPT_POST, 1); // 添加post數據到請求中 curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data); // 執行post請求,得到回覆 $response= curl_exec($ch); curl_close($ch); echo $response;以上請求發送到post.php中後,經過print_r($_POST)輸出後,以上示例代碼會輸出以下回復:
Array ( [blog_name] => 52fhy [blog_url] => http://www.52fhy.com [action] => Submit )
正如咱們看到的,cURL成功發送post請求到post.php,提交了一些數據,而且收到了相應的來自post.php的回覆,最後輸出回覆。上例雖然簡單,可是充分演示了cURL發送post請求的便捷及強大之處,你能夠在curl_setopt上作文章。
下面來看下若是經過cURL發送post請求來實現文件上傳。就拿深刻淺出PHP下的文件上傳中的文件上傳例子來演示,在深刻淺出php下的文件上傳中,是經過表單的提交來實現文件上傳的,那麼經過cURL怎麼來實現呢?
$url = "http://www.52fhy.me/upload.php"; $post_data = array ( "attachment" => "@E:/jackblog/boy.jpg" ); //初始化cURL會話 $ch = curl_init(); //設置請求的url curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); //設置爲post請求類型 curl_setopt($ch, CURLOPT_POST, 1); //設置具體的post數據 curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data); $response = curl_exec($ch); curl_close($ch); print_r($response);經過以上示例代碼,能夠將我本地機器上的boy.jpg上傳到本地服務器的upload.php中,若是在upload.php輸出上傳的具體信息的話,以上示例代碼最後的輸出的回覆爲:
Array ( [attachment] => Array ( [name] => boy.jpg [type] => application/octet-stream [tmp_name] => D:\xampp\tmp\phpF27D.tmp [error] => 0 [size] => 11490 ) )
因而可知,若是你要經過cURL來上傳文件的話,只須要將上傳的文件路徑做爲post數據設置到curl請求中,而且在路徑前面加上@符合。
上述將了文件上傳,一樣的也可使用curl來自動地完成文件的下載以及保存。有一點要補充下,在執行一個curl請求時,若是你須要獲取返回的內容,而不是直接輸出返回的內容的話,別忘記使用下面的代碼設置,由於curl的默認是輸出請求的回覆內容:
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
假如在52fhy的服務器根目錄下面有一個test.zip文件,咱們須要將其下載下來,而且保存到本地文件中,就能夠嘗試使用下面的代碼來實現:
//設置請求的下載文件的url $url = 'http://www.52fhy.com/test.zip'; //保存到本地的文件路徑 $path = 'local/path/to/test.zip'; //初始化請求,設置請求,獲取回覆,關閉會話 $ch = curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $data = curl_exec($ch); curl_close($ch); //將文件內容寫入本地文件 file_put_contents($path, $data);
注意:我以上省略了錯誤處理方面的代碼,只是簡單作個示例, 在實際中,你還須要經過curl_getinfo
函數來進行錯誤處理!
上述代碼對於下載比較大型的文件是不適用的,由於須要先將文件讀取到內存中,等全部內容都讀取完畢,而後再寫入到本地硬盤中。即便php中設置的 memory limit很是大,這種狀況對性能的影響也是很大的。因此,咱們對於大型文件的下載,應該讓curl來接管這個任務,實現邊下載,邊寫入的處理,這樣的 話,就沒什麼問題了。請看下述代碼:
$url = 'http://www.52fhy.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);
在上述代碼中,咱們先打開個本地文件,並將文件句柄設置到curl中,而後讓curl一邊讀取遠程數據,一邊寫入到本地文件中。由於咱們不須要在程序中獲取遠程回覆的內容了,因此只要執行請求就能夠。
若是服務器端須要驗證請求,能夠經過相似一下示例代碼來實現:
$url = "http://www.52fhy.com/users/"; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設置用戶名以及密碼 curl_setopt($ch, CURLOPT_USERPWD, "username:password"); // 設置重導向 curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1); curl_setopt($ch, CURLOPT_UNRESTRICTED_AUTH, 1); $response = curl_exec($ch); curl_close($ch);
cURL還能夠經過代理服務器來向發送請求,請看一下示例代碼:
$ch = curl_init(); curl_setopt($ch, CURLOPT_URL,'http://www.52fhy.com'); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 設置代理ip地址 curl_setopt($ch, CURLOPT_PROXY, '222.73.173.50:8080'); // 要驗證的話,這裏設置用戶名以及密碼 curl_setopt($ch, CURLOPT_PROXYUSERPWD,'username:password'); $response = curl_exec($ch); curl_close ($ch);
最後,咱們來看下經過cURL來想服務器端發送json數據。具體的代碼以下:
$url = 'http://www.52fhy.me/json.php'; // 創建json字符串 $data = array('site' => '52fhy', 'url' => 'http://www.52fhy.com','email'=>'52fhy@gmail.com'); $json_string = json_encode($data); $ch=curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 經過post請求發送上述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;你們能夠看到,上述請求是發送到個人本地服務器的json.php下,我在該文件中使用json_decode來將接受到的json字符串轉換爲對象,而後輸出其中的email字段,代碼以下:
$json_data = json_decode($_POST['data']); echo $json_data->email;
'{"site":"52fhy","url":"http:\/\/www.52fhy.com","email":"52fhy@gmail.com"}'
通過json_decode之後,就轉換爲php中的數據格式,成爲了一個對象,因此能夠經過$json_data->email
來訪問其中email字段的值,最後也就是輸出52fhy@gmail.com。你可使用上述代碼測試一下。
若是經過如下php數組生成json字符串的話:
$data = array('52fhy', 'http://www.52fhy.com', '52fhy@gmail.com');所生成的json字符串以下:
'["52fhy","http:\/\/www.52fhy.com","52fhy@gmail.com"]'
上述json字符串在通過json_decode處理後,就會變成php中的數組格式,若是要獲取email的話,就能夠經過$json_data[2]
來訪問。
cURL還有一個高級特性——批處理句柄(handle)。這一特性容許你同時或異步地打開多個URL鏈接。
下面是來自來自php.net的示例代碼:
// 建立兩個cURL資源 $ch1 = curl_init(); $ch2 = curl_init(); // 指定URL和適當的參數 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); // 建立cURL批處理句柄 $mh = curl_multi_init(); // 加上前面兩個資源句柄 curl_multi_add_handle($mh,$ch1); curl_multi_add_handle($mh,$ch2); // 預約義一個狀態變量 $active = null; // 執行批處理 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); } } // 關閉各個句柄 curl_multi_remove_handle($mh, $ch1); curl_multi_remove_handle($mh, $ch2); curl_multi_close($mh);
這裏要作的就是打開多個cURL句柄並指派給一個批處理句柄。而後你就只需在一個while循環裏等它執行完畢。
這個示例中有兩個主要循環。第一個 do-while 循環重複調用 curl_multi_exec()
。這個函數是無隔斷(non-blocking)的,但會盡量少地執行。它返回一個狀態值,只要這個值等於常量 CURLM_CALL_MULTI_PERFORM
,就表明還有一些刻不容緩的工做要作(例如,把對應URL的http頭信息發送出去)。也就是說,咱們須要不斷調用該函數,直到返回值發生改變。
而接下來的 while 循環,只在 $activ
e 變量爲 true 時繼續。這一變量以前做爲第二個參數傳給了 curl_multi_exec()
,表明只要批處理句柄中是否還有活動鏈接。接着,咱們調用 curl_multi_select()
,在活動鏈接(例如接受服務器響應)出現以前,它都是被「屏蔽」的。這個函數成功執行後,咱們又會進入另外一個 do-while 循環,繼續下一條URL。
在這篇博文中只是列舉了一些cURL的用途,其中示例代碼是比較簡單的。可是,相信你看完後應該有使用cURL的衝動了吧! 那就本身去找相關資料,手冊進行測試吧!
好了,就寫到這裏吧!謝謝你的耐心閱讀!
附:
<?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請求中包含一個"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); // 對認證證書來源的檢查 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; } }