欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

PHP實(shí)現(xiàn)文件下載斷點(diǎn)續(xù)傳詳解

 更新時(shí)間:2014年10月15日 13:36:45   投稿:junjie  
這篇文章主要介紹了PHP實(shí)現(xiàn)文件下載斷點(diǎn)續(xù)傳詳解,本文講解了載斷點(diǎn)續(xù)傳的實(shí)現(xiàn)理解,并給出了實(shí)現(xiàn)代碼,需要的朋友可以參考下

如果我們的網(wǎng)站提供文件下載的服務(wù),那么通常我們都希望下載可以斷點(diǎn)續(xù)傳(Resumable Download),也就是說用戶可以暫停下載,并在未來的某個(gè)時(shí)間從暫停處繼續(xù)下載,而不必重新下載整個(gè)文件。

通常情況下,Web服務(wù)器(如Apache)會(huì)默認(rèn)開啟對(duì)斷點(diǎn)續(xù)傳的支持。因此,如果直接通過Web服務(wù)器來提供文件的下載,可以不必做特別的配置,即可享受到斷點(diǎn)續(xù)傳的好處。由于這些文件直接通過Web服務(wù)器來提供下載,后端腳本無法對(duì)這個(gè)下載過程進(jìn)行控制。這對(duì)于僅提供公開、靜態(tài)文件的網(wǎng)站來說不是問題,但對(duì)于需要提供私有、動(dòng)態(tài)文件的網(wǎng)站來說,直接通過Web服務(wù)器來提供下載就無法滿足需求了。這時(shí),就需要在編寫后臺(tái)腳本程序時(shí),加入對(duì)斷點(diǎn)續(xù)傳的支持。

本文將以PHP為例,簡要介紹實(shí)現(xiàn)文件下載斷點(diǎn)續(xù)傳的方法。

原理

斷點(diǎn)續(xù)傳的原理還是比較直觀的。

HTTP協(xié)議規(guī)定了如何傳輸某個(gè)資源的一部分,而不是全部。比如,有一個(gè)文件的大小是1000字節(jié),瀏覽器可以只請(qǐng)求該文件的前300個(gè)字節(jié),或者只請(qǐng)求第500到第1000個(gè)字節(jié)。通過這種方式,就可以不必在一次請(qǐng)求中傳輸某個(gè)資源的全部內(nèi)容,而是發(fā)起多次請(qǐng)求,每次僅請(qǐng)求其中的一部分內(nèi)容。等所有這些請(qǐng)求都返回之后,再把得到的內(nèi)容一塊一塊的拼接起來得到完整的資源。

實(shí)現(xiàn)斷點(diǎn)續(xù)傳就是要利用HTTP協(xié)議的上述特性。當(dāng)用戶暫停下載的時(shí)候,瀏覽器會(huì)記錄已經(jīng)下載到什么位置,當(dāng)用戶在未來某一時(shí)間恢復(fù)下載時(shí),就可以從上次暫停的位置繼續(xù)下載,而不必從頭開始。

實(shí)現(xiàn)

由于部分傳輸不是強(qiáng)制的,服務(wù)器可以支持也可以不支持,所以,我們需要在程序中告訴瀏覽器,它請(qǐng)求的資源是否支持部分傳輸。這可以通過設(shè)置HTTP的 Accept-Ranges 響應(yīng)頭信息來實(shí)現(xiàn)。PHP代碼如下:

復(fù)制代碼 代碼如下:

header('Accept-Ranges: bytes');

Accept-Ranges: bytes 告訴瀏覽器,該資源支持以字節(jié)為單位的部分傳輸。這個(gè)響應(yīng)頭需要附加在支持部分傳輸?shù)乃匈Y源上。

當(dāng)接受到一個(gè)請(qǐng)求時(shí),我們需要從瀏覽器的請(qǐng)求中提取瀏覽器具體是在請(qǐng)求資源的哪一個(gè)部分。這個(gè)信息是通過 Range 請(qǐng)求頭來傳遞的。在PHP中,它被存儲(chǔ)在$_SERVER['HTTP_RANGE']中。我們需要檢查這個(gè)變量是否定義了,如果定義了,則使用該值,否則,就將range設(shè)為整個(gè)資源。

復(fù)制代碼 代碼如下:

$range = "0-". ($content_length-1);
if(isset($_SERVER['HTTP_RANGE'])){
    $range = $_SERVER['HTTP_RANGE'];
}

接下來,就需要分析 $range 的值,來決定返回資源的哪一部分內(nèi)容??赡艿娜≈凳纠?br />
復(fù)制代碼 代碼如下:

100-200 // 第100到第200字節(jié)
500-    // 第500字節(jié)到文件末尾
-1000   // 最后的1000個(gè)字節(jié)

這里需要注意,得到一個(gè)Range之后,你需要對(duì)它的取值進(jìn)行檢驗(yàn),包括:

1.開始位置非負(fù)
2.結(jié)束位置需要大于開始位置
3.開始位置需要小于文件長度減一 (因?yàn)檫@里的位置索引是從0開始的)
4.若結(jié)束位置大于文件長度減一,則需要把它的值設(shè)置為文件長度減一

如果Range的取值不合法,則需要終止程序并告知瀏覽器:

復(fù)制代碼 代碼如下:

header('HTTP/1.1 416 Requested Range Not Satisfiable');

為了保持文章簡潔,具體的校驗(yàn)代碼這里就不提供了。下面假定你已經(jīng)校驗(yàn)了Range的取值,并得到了 $start 和 $end 兩個(gè)變量,分別表示開始位置和結(jié)束位置。

接下來要做的就是把文件的對(duì)應(yīng)部分的內(nèi)容發(fā)送給瀏覽器。不過要注意的是,這里涉及到需要發(fā)送多個(gè)HTTP響應(yīng)頭信息,具體如下:

復(fù)制代碼 代碼如下:

header('HTTP/1.1 206 Partial Content');
header('Accept-Ranges: bytes');
header("Content-Range: bytes $start-$end/$filesize");
$length = $end - $start + 1;
header("Content-Length: $length");

/* 輸出文件的指定部分 */


這里的$length需要注意一下,它的取值是本次傳輸?shù)膬?nèi)容的長度,而不是整個(gè)文件的長度。另外需要注意的一點(diǎn)是,這里的HTTP狀態(tài)碼是206,不是200。

總結(jié)

文件下載的斷點(diǎn)續(xù)傳實(shí)際上是利用了HTTP協(xié)議中對(duì)傳輸部分文件的支持。而HTTP協(xié)議的這一特性不僅可以用于實(shí)現(xiàn)斷點(diǎn)續(xù)傳,客戶端程序也可以利用它來實(shí)現(xiàn)多線程下載。

在實(shí)現(xiàn)斷點(diǎn)續(xù)傳的過程中,需要注意正確設(shè)置各種HTTP頭信息。錯(cuò)誤的頭信息將導(dǎo)致用戶下載到的文件損壞,無法使用。

相關(guān)文章

最新評(píng)論