1. 程式人生 > >php中的curl使用入門教程和常見用法實例(轉載)

php中的curl使用入門教程和常見用法實例(轉載)

memory stat 腳本 rem esp 增加 enc strip email

摘自:https://www.cnblogs.com/echohao/p/6134673.html

摘要: [目錄] php中的curl使用入門教程和常見用法實例 一、curl的優勢 二、curl的簡單使用步驟 三、錯誤處理 四、獲取curl請求的具體信息 五、使用curl發送post請求 六、文件上傳 七、文件下載 八、http 驗證 九、通過代理發送請求 十、發送json數據 十一、cURL批處理(...


[目錄]
php中的curl使用入門教程和常見用法實例
一、curl的優勢
二、curl的簡單使用步驟
三、錯誤處理
四、獲取curl請求的具體信息
五、使用curl發送post請求
六、文件上傳
七、文件下載
八、http 驗證
九、通過代理發送請求
十、發送json數據
十一、cURL批處理(multi cURL)
十二、總結

起先cURL是做為一種命令行工具設計出來的,比較幸運的是,php也支持cURL了。通過cURL這個利器,我們能在php程序中自由地發送 HTTP請求到某個url來獲取或者提交數據,並且支持其它多種協議,比如FTP,Telnet以及SMTP等。在這篇博文中,我將簡述下,在php中具 體怎麽使用cURL來處理一些事情。

一、curl的優勢

你也許會說,在php中可以很容易的獲取某個url的內容,只要通過file_get_contents,file或者readfile函數就能輕松實現,根本不必使用cURL:

$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的簡單使用步驟

要使用cURL來發送url請求,具體步驟大體分為以下四步:

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.php

三、錯誤處理

在上述代碼中,你也可以增加錯誤處理的代碼:

$response = curl_exec($ch);

if ($response  === FALSE) {
  echo "cURL 具體出錯信息: " . curl_error($ch);
}

註意了,在做上述判斷時務必要使用===,因為請求的回復可能是空字符串,curl在請求出錯的情況下回返回FALSE值,所以我們必須使用===,而不是==

四、獲取curl請求的具體信息

在執行一個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

五、使用curl發送post請求

我們在前面說過,在向某個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一邊讀取遠程數據,一邊寫入到本地文件中。因為我們不需要在程序中獲取遠程回復的內容了,所以只要執行請求就可以。

八、http 驗證

如果服務器端需要驗證請求,可以通過類似一下示例代碼來實現:

$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);

十、發送json數據

最後,我們來看下通過cURL來想服務器端發送json數據。具體的代碼如下:

$url = ‘http://www.52fhy.me/json.php‘;

// 建立json字符串
$data = array(‘site‘ => ‘52fhy‘, ‘url‘ => ‘http://www.52fhy.com‘,‘email‘=>‘[email protected]‘);
$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;

在上述代碼中接受的json字符串為:
‘{"site":"52fhy","url":"http:\/\/www.52fhy.com","email":"[email protected]"}‘

經過json_decode以後,就轉換為php中的數據格式,成為了一個對象,所以可以通過$json_data->email來訪問其中email字段的值,最後也就是輸出[email protected]。你可以使用上述代碼測試一下。

如果通過以下php數組生成json字符串的話:

$data = array(‘52fhy‘, ‘http://www.52fhy.com‘, ‘[email protected]‘);
所生成的json字符串如下:
‘["52fhy","http:\/\/www.52fhy.com","[email protected]"]‘

上述json字符串在經過json_decode處理後,就會變成php中的數組格式,如果要獲取email的話,就可以通過$json_data[2]來訪問。

十一、cURL批處理(multi cURL)

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 循環,只在 $active 變量為 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;
    }
}

php中的curl使用入門教程和常見用法實例(轉載)