所以根據EOF就可判斷一次請求的結束,下面的代碼(PHP)很常見:
復制代碼 代碼如下:
// $fp是由fsockopen()產生的句柄
while(!feof($fp)) {
echo fgets($fp);
}
(注:短連接模式是在頭部用”Connection: close”標示,長連接用”Connection: keep-alive”標示。目前HTTP/1.0默認使用短連接,HTTP/1.1默認使用長連接。)
而長連接(也稱持久連接)模式的HTTP在發送完數據後服務器並不斷開連接,而是留著下一次HTTP請求時使用,所以長連接的好處是顯而易見的,通過共用一個TCP連接來節省以後請求時建立/斷開連接的開銷。而EOF是直到這個TCP連接結束(超時或出錯)時才會被發送,所以我們就不能使用上面的辦法來判斷一次HTTP請求的結束了。這也是使用長連接時都會遇到的一個問題。目前判斷的方法主要有兩種:
(1) 根據頭中的Content-Length字段。這個字段標明了正文的長度,我們可以以接收完指定長度的字符為判斷結束的依據。
(2) 在沒有Content-Length時,根據Transfer-Encoding。有些時候服務器無法確定正文的大小,因為正文可能是動態產生的,所以就不會提供Content-Length了,而是采用chunk編碼來一塊一塊地發送正文。每個chunk塊由頭部和正文兩部分組成,頭部中由一個16進制數字指定了正文的長度;最後由一個長度為0的chunk塊來表示整個HTTP正文的結束。
下面我用PHP實現了有Content-Length時的判斷方式:
1. 獲得Content-Length值
復制代碼 代碼如下:
$length = 0;
$line = '';
while($line !== "\r\n") {
$line = fgets($fp);
if(substr($line, 0, 15) === 'Content-Length:') {
$length = intval(substr($line, 16));
}
}
2. 獲得正文
復制代碼 代碼如下:
$sum = 0;
while($sum < $length) {
$line = fgets($fp);
$sum += strlen($line);
echo $line;
}