明輝手游網(wǎng)中心:是一個免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺!

在ASP.NET中跟蹤與恢復(fù)大文件下載

[摘要]文/陶剛  在Web應(yīng)用程序中處理大文件下載的問題一直出了名的困難,因此對于大多數(shù)站點(diǎn)來說,如果用戶的下載被中斷了,它們只能說悲哀降臨到用戶的身上了。但是我們現(xiàn)在不必這樣了,因?yàn)槟憧梢允棺约旱腁SP.NET應(yīng)用程序有能力支持可恢復(fù)(繼續(xù))的大文件下載。使用本文提供的方法的時候,你可以跟蹤下載的過程...

  文/陶剛

  在Web應(yīng)用程序中處理大文件下載的問題一直出了名的困難,因此對于大多數(shù)站點(diǎn)來說,如果用戶的下載被中斷了,它們只能說悲哀降臨到用戶的身上了。但是我們現(xiàn)在不必這樣了,因?yàn)槟憧梢允棺约旱腁SP.NET應(yīng)用程序有能力支持可恢復(fù)(繼續(xù))的大文件下載。使用本文提供的方法的時候,你可以跟蹤下載的過程,這樣你就可以處理動態(tài)建立的文件--而且要達(dá)到這個目標(biāo)根本不需要舊式的ISAPI動態(tài)鏈接庫和非受控的(unmanaged)C++代碼。

  為客戶端提供從互聯(lián)網(wǎng)上下載文件的服務(wù)最容易了,對嗎?僅僅只需要把可下載的文件復(fù)制到你的Web應(yīng)用程序目錄中,發(fā)布鏈接并讓IIS完成所有相關(guān)的工作。但是,文件服務(wù)不應(yīng)該比脖子上的疼痛還要多(還要麻煩),你不希望整個世界都能訪問自己的數(shù)據(jù),你不希望服務(wù)器被數(shù)百個靜態(tài)文件塞滿了,你甚至于希望下載臨時文件--只有當(dāng)客戶端開始下載后的空閑時間才建立這些文件。

  不幸的是,使用IIS對下載請求的默認(rèn)的響應(yīng)是不可能達(dá)到這些效果的。因此在一般情況下,為了獲得對下載過程的控制權(quán),開發(fā)者需要鏈接到一個定制的.aspx頁面,在這個頁面中它們檢查用戶憑證(credential)、建立可以下載的文件并使用下面的代碼把該文件推送給客戶端:

Response.WriteFile
Response.End()

  而這就是出現(xiàn)真正麻煩的地方。

  有什么問題?

  WriteFile方法看起來非常完美,它使文件的二進(jìn)制數(shù)據(jù)流向客戶端。但是直到最近我們才知道,WriteFile方法是一個出名的內(nèi)存占用狂,它把整個文件載入服務(wù)器的RAM中來提供服務(wù)(實(shí)際上它甚至于會占用文件兩倍大小的空間)。對于大文件,這會引起服務(wù)內(nèi)存問題,并且可能重復(fù)ASP.NET過程。但是在2004年6月微軟發(fā)布了一個補(bǔ)丁解決了這個問題。這個補(bǔ)丁現(xiàn)在是.NET Framework 1.1補(bǔ)丁包(SP1)的一部分。

  這個補(bǔ)丁引入了TransmitFile方法,它把一個磁盤文件讀入到較小的內(nèi)存緩沖區(qū)之后就開始傳輸該文件。盡管這個方案解決了內(nèi)存和循環(huán)的問題,但是它仍然不能令人滿意。你不能控制響應(yīng)的生命周期。你無法知道下載是否正確地完成了,你沒有辦法知道下載是否被中斷了,并且(如果你建立了臨時文件)你也不知道是否應(yīng)該、以及什么時候可以刪除這些文件。更糟的是,如果下載的確失敗了,TransmitFile方法又從客戶端下次嘗試的文件頭部開始下載。

  其中一種可能的解決方案--實(shí)現(xiàn)后臺智能傳輸服務(wù)(BITS)對于多數(shù)站點(diǎn)來說是不可行的,因?yàn)檫@會毀掉維持客戶端瀏覽器和操作系統(tǒng)獨(dú)立性而作出的努力。

  令人滿意的解決方案的基礎(chǔ)還是來自微軟用于解決WriteFile引起的內(nèi)存混亂問題的第一次嘗試(見知識庫文章812406)。那篇文章演示了智能的大塊數(shù)據(jù)下載過程,它從文件流中讀取數(shù)據(jù)。在服務(wù)器把字節(jié)塊發(fā)送給客戶端之前,它使用Response.IsClientConnected屬性檢查客戶端是否仍然保持著連接。如果仍然保持連接,它就繼續(xù)發(fā)送流字節(jié),否則就停止,以防止服務(wù)器發(fā)送不必要的數(shù)據(jù)。
這就是我們采用的方法,特別是在下載臨時文件的時候。在IsClientConnected返回False的情況下,你就知道下載過程被中斷了,你應(yīng)該保存文件;反之,當(dāng)這個過程成功完成的時候,你就刪除臨時文件。此外,為了恢復(fù)中斷了的下載,你需要做的工作是從上次下載嘗試過程中客戶端連接失敗的文件點(diǎn)開始下載。

  HTTP協(xié)議和頭信息(Header)支持

  HTTP協(xié)議支持可以用于處理被中斷下載的頭信息。使用少量的HTTP頭信息,你可以增強(qiáng)自己的下載過程,使它完全遵循HTTP協(xié)議規(guī)范。這個規(guī)范與ranges一起提供恢復(fù)被中斷的下載所需要的一切信息。

  下面是它的工作方式。首先,如果服務(wù)器支持客戶端斷點(diǎn)續(xù)傳,它就在初始的響應(yīng)中發(fā)送Accept-Ranges頭信息。服務(wù)器還發(fā)送一個實(shí)體標(biāo)簽(entity tag)頭信息(ETag),它包含一個唯一的標(biāo)識字符串。

  下面的代碼顯示了IIS發(fā)送給客戶端的用于響應(yīng)一個初始下載請求的一些頭信息,它向客戶端傳遞了被請求的文件的詳細(xì)信息。

HTTP/1.1 200 OK
Connection: close
Date: Tue, 19 Oct 2004 15:11:23 GMT
Accept-Ranges: bytes
Last-Modified: Sun, 26 Sep 2004 15:52:45 GMT
ETag: "47febb2cfd76c41:2062"
Cache-Control: private
Content-Type: application/x-zip-compressed
Content-Length: 2844011

  在接收這些頭信息之后,如果下載被中斷了,IE瀏覽器在后來的下載請求中會把Etag值和Range頭信息發(fā)送回服務(wù)器。下面的代碼顯示了嘗試恢復(fù)被中斷下載時IE發(fā)送給服務(wù)器的一些頭信息。

GET http://192.168.100.100/download.zip HTTP/1.0
Range: bytes=822603-
Unless-Modified-Since: Sun, 26 Sep 2004 15:52:45 GMT
If-Range: "47febb2cfd76c41:2062"

  這些頭信息表明IE緩存了IIS提供的實(shí)體標(biāo)簽,并在If-Range頭信息中把它發(fā)送回服務(wù)器了,這是確保下載從準(zhǔn)確相同的文件恢復(fù)的一種途徑。不幸的是,并非所有的瀏覽器的工作方式都相同?蛻舳税l(fā)送的用于驗(yàn)證文件的其它HTTP頭信息可能是If-Match、If-Unmodified-Since或者Unless-Modified-Since。很明顯,該規(guī)范對于客戶端軟件必須支持哪些頭信息,或者必須使用哪些頭信息沒有明確的規(guī)定。因此,有些客戶端根本就沒有使用頭信息,而IE只使用If-Range和Unless-Modified-Since。你最好用代碼檢查這些信息。采用這種方式的時候,你的應(yīng)用程序可以在非常高的層次遵循HTTP規(guī)范,并可以使用多種瀏覽器。Range頭信息指明了被請求的字節(jié)范圍--在例子中它是服務(wù)器應(yīng)該恢復(fù)文件流的起始點(diǎn)。

  當(dāng)IIS接收到恢復(fù)下載的請求類型時,它發(fā)回包含下面的頭信息的響應(yīng)信息:

HTTP/1.1 206 Partial Content
Content-Range: bytes 822603-2844010/2844011
Accept-Ranges: bytes
Last-Modified: Sun, 26 Sep 2004 15:52:45 GMT
ETag: "47febb2cfd76c41:2062"
Cache-Control: private
Content-Type: application/x-zip-compressed
Content-Length: 2021408


  請注意上面的代碼與最初的下載請求的HTTP響應(yīng)有點(diǎn)差別--恢復(fù)下載的請求是206而最初下載的請求是200。這表明通過線路傳遞進(jìn)來的內(nèi)容是部分文件。這一次Content-Range頭信息指出了被傳遞字節(jié)的精確數(shù)量和位置。

  IE對于這些頭信息是很挑剔的。如果最初的響應(yīng)沒有包含Etag頭信息,IE永遠(yuǎn)不會嘗試恢復(fù)下載。我測試過的其它客戶端不使用ETag頭信息,它們簡單得依賴于文件名、請求范圍,并使用Last-Modified頭信息(如果它們試圖驗(yàn)證該文件)。

  深入了解HTTP協(xié)議

  前面的部分中顯示的頭信息對于使恢復(fù)下載的解決方案運(yùn)行來說是足夠的,但是它沒有完全覆蓋HTTP規(guī)范。

  在單個請求中,Range頭信息可以詢問多個范圍,這種特性稱為"多部分范圍(multipart ranges)"。請不要與分段下載(segmented downloading)混淆,幾乎所有的下載工具都使用分段下載來提高下載速度。這些工具聲稱通過打開兩個或多個并發(fā)的連接(每個連接請求文件的不同范圍)提高了下載速度。

  多部分范圍的想法并沒有開啟多個連接,但是它可以使客戶端軟件可以在單個請求/響應(yīng)周期中請求某個文件的最前面的十個和最后面的十個字節(jié)。

  誠實(shí)地說,我從來都沒有找到使用這種特性軟件片斷。但是我拒絕在代碼聲明中寫入"它并不是完全的HTTP兼容的"。略去這個特性必定會觸犯墨菲法則(Murphy's Law)。無論如何,多部分范圍還是被用于電子郵件傳輸中,把頭信息、普通文本和附件分開。

  示例代碼

  我們知道了客戶端和服務(wù)器如何交換頭信息以保證可恢復(fù)的下載,把這些知識與文件塊流的思想結(jié)合起來,你就可以給自己的ASP.NET應(yīng)用程序增加可靠的下載管理能力了。

  獲取下載過程的控制權(quán)的方法是從客戶端截取下載請求、讀取頭信息并適當(dāng)?shù)仨憫?yīng)。在.NET之前,你必須編寫ISAPI(Internet服務(wù)器API)應(yīng)用程序來實(shí)現(xiàn)這種功能,但是.NET框架組件提供了一個IHttpHandler接口,在類中實(shí)現(xiàn)的時候,它允許你僅僅使用.NET代碼就能夠截取和處理請求。這意味著你的應(yīng)用程序?qū)τ谙螺d過程有完全控制權(quán)和響應(yīng)性,再也不會涉及或使用IIS的自動化函數(shù)。

  示例代碼在HttpHandler.vb文件中包含了一個自定義的HttpHandler類(ZIPHandler)。ZipHandler實(shí)現(xiàn)了IhttpHandler接口,并且處理對所有.zip文件的請求。

  為了測試示例代碼,你需要在IIS中建立一個新的虛擬目錄,并把源文件復(fù)制到那兒。在該目錄中建立一個叫做download.zip的文件(請注意IIS和ASP.NET不能處理大于2GB的下載,因此要確保你的文件沒有超過該限制)。配置你的IIS虛擬目錄,通過aspnet_isapi.dll映射.zip擴(kuò)展名。

  HttpHandler類:ZIPHandler

  在ASP.NET中映射了.zip擴(kuò)展名之后,客戶端每次向服務(wù)器請求.zip文件的時候,IIS調(diào)用ZipHandler類的ProcessRequest方法(見下載代碼)。

  ProcessRequest方法首先建立自定義的FileInformation類(見下載代碼)的一個實(shí)例,它封裝了下載的狀態(tài)(例如進(jìn)行中、被中斷了等等)。示例把download.zip示例文件的路徑硬編碼到代碼中了。如果把這段代碼應(yīng)用于你自己的應(yīng)用程序,需要修改它來打開被請求的文件。

 

' 使用objRequest檢測請求了哪個文件,用該文件打開objFile。
' 例如objFile = New Download.FileInformation(<完整文件名>)
objFile = New Download.FileInformation( _
objContext.Server.MapPath("~/download.zip"))

 

  接下來,程序使用描述的HTTP頭信息(如果請求提供了頭信息)執(zhí)幸幌盜械難櫓ぜ觳欏K衙恐旨觳槎擠庾霸諦⌒退接瀉,染J櫓こ曬Φ幕熬頭禱豑rue。如果某個驗(yàn)證檢查失敗了,響應(yīng)會立即終止,并發(fā)送適當(dāng)?shù)腟tatusCode值。

 

If Not objRequest.HttpMethod.Equals(HTTP_METHOD_GET) Or Not
objRequest.HttpMethod.Equals(HTTP_METHOD_HEAD) Then
 ' 目前只支持GET和HEAD方法
 objResponse.StatusCode = 501 ' 沒有執(zhí)行
ElseIf Not objFile.Exists Then
 ' 無法找到被請求的文件
 objResponse.StatusCode = 404 ' 沒有找到
ElseIf objFile.Length > Int32.MaxValue Then
 ' 文件太大了
 objResponse.StatusCode = 413 ' 請求實(shí)體太大
ElseIf Not ParseRequestHeaderRange(objRequest, alRequestedRangesBegin, alRequestedRangesend, _
objFile.Length, bIsRangeRequest) Then
 ' Range請求中包含無用的實(shí)體
 objResponse.StatusCode = 400 ' 無用的請求
ElseIf Not CheckIfModifiedSince(objRequest,objFile) Then
 ' 實(shí)體沒有被修改過
 objResponse.StatusCode = 304 ' 沒有被修改過
ElseIf Not CheckIfUnmodifiedSince(objRequest,objFile) Then
 ' 實(shí)體在上次被請求的日期之后被修改過
 objResponse.StatusCode = 412 ' 預(yù)處理失敗
ElseIf Not CheckIfMatch(objRequest, objFile) Then
 ' 實(shí)體與請求不匹配
 objResponse.StatusCode = 412 ' 預(yù)處理失敗
ElseIf Not CheckIfNoneMatch(objRequest, objResponse,objFile) Then
 ' 實(shí)體的確與none-match請求匹配。
 ' 響應(yīng)代碼位于CheckIfNoneMatch函數(shù)中
Else
 ' 初步檢查成功

 

  這些初步檢查的函數(shù)中的ParseRequestHeaderRange(見下載代碼)檢查客戶端是否請求了文件范圍(這意味著是一個局部下載)。如果被請求的范圍是無效的(無效范圍指超越文件大小或包含不合理數(shù)字的范圍數(shù)值),該方法把bIsRangeRequest設(shè)置為True。如果請求了范圍,CheckIfRange方法會驗(yàn)證IfRange頭信息。

  如果被請求的范圍是有效的,代碼會計算響應(yīng)信息的大小。如果客戶端請求了多個范圍,響應(yīng)信息大小的數(shù)值會包含多部分頭部信息長度的數(shù)值。

  如果不能確定某個發(fā)送的頭部信息值,程序?qū)堰@個下載請求作為最初請求而不是部分下載來處理,從文件的頂部開始發(fā)送一個新的下載流。

 

If bIsRangeRequest AndAlso CheckIfRange(objRequest, objFile) Then
 ' 這是范圍請求
 ' 如果Range數(shù)組包含多個實(shí)體,它還是一個多部分范圍請求
 bMultipart = CBool(alRequestedRangesBegin.GetUpperBound(0)>0)
 ' 進(jìn)入每個范圍來獲取整個響應(yīng)長度
 For iLoop = alRequestedRangesBegin.GetLowerBound(0) To alRequestedRangesBegin.GetUpperBound(0)
  ' 內(nèi)容的長度(這個范圍的)
  iResponseContentLength += Convert.ToInt32(alRequestedRangesend( _
iLoop) - alRequestedRangesBegin(iLoop)) + 1
  If bMultipart Then
   ' 如果是多部分范圍請求,計算出將發(fā)送的中間頭信息的長度
   iResponseContentLength += MULTIPART_BOUNDARY.Length
   iResponseContentLength += objFile.ContentType.Length
   iResponseContentLength += alRequestedRangesBegin(iLoop).ToString.Length
   iResponseContentLength += alRequestedRangesend(iLoop).ToString.Length
   iResponseContentLength += objFile.Length.ToString.Length
   ' 49是多部分下載中換行和其它必要的字符的長度
   iResponseContentLength += 49
  End If
 Next iLoop

 If bMultipart Then
  ' 如果是多部分范圍請求,
  ' 我們還必須計算出將發(fā)送的最后一個中間頭信息的長度
  iResponseContentLength +=MULTIPART_BOUNDARY.Length
  ' 8 是破折號和換行符的長度
  iResponseContentLength += 8
 Else
  ' 不是多部分下載,因此我們必須說明初始HTTP頭信息的響應(yīng)范圍
  objResponse.AppendHeader( HTTP_HEADER_CONTENT_RANGE, "bytes " & _
  alRequestedRangesBegin(0).ToString & "-" & _
  alRequestedRangesend(0).ToString & "/" & _
  objFile.Length.ToString)
  'End If
  ' 范圍響應(yīng)
  objResponse.StatusCode = 206 ' 局部響應(yīng)
 Else
  ' 這不是范圍請求,或者被請求的范圍實(shí)體ID與當(dāng)前的實(shí)體ID不匹配,
  ' 因此開始新的下載
  ' 指明文件完成部分的大小等于內(nèi)容的長度
  iResponseContentLength =Convert.ToInt32(objFile.Length)
  ' 返回正常的OK狀態(tài)
  objResponse.StatusCode = 200
 End If
 ' 接下來服務(wù)器必須發(fā)送幾個重要的響應(yīng)頭信息,例如內(nèi)容長度、Etag、和文件的內(nèi)容類型:
 ' 把內(nèi)容長度寫入響應(yīng)
 objResponse.AppendHeader( HTTP_HEADER_CONTENT_LENGTH,iResponseContentLength.ToString)
 ' 把最后修改日期寫入響應(yīng)
 objResponse.AppendHeader( HTTP_HEADER_LAST_MODIFIED,objFile.LastWriteTimeUTC.ToString("r"))
 ' 告訴客戶端軟件我們接受了范圍請求
 objResponse.AppendHeader( HTTP_HEADER_ACCEPT_RANGES,HTTP_HEADER_ACCEPT_RANGES_BYTES)
 ' 把文件的實(shí)體標(biāo)簽寫入響應(yīng)(用引號括起來)
 objResponse.AppendHeader(HTTP_HEADER_ENTITY_TAG, """" & objFile.EntityTag & """")
 ' 把內(nèi)容類型寫入響應(yīng)
 If bMultipart Then
  ' 多部分消息有這種特殊的類型
  ' 在例子中文件實(shí)際的mime類型在以后才寫入響應(yīng)
  objResponse.ContentType = MULTIPART_CONTENTTYPE
 Else
  ' 單個部分消息擁有的文件內(nèi)容類型
  objResponse.ContentType = objFile.ContentType
End If

 


  下載所需要的一切都準(zhǔn)備好了,可以開始下載文件了。你將使用FileStream對象從文件中讀取字節(jié)塊。把FileInformation實(shí)例objFile的State屬性設(shè)置為fsDownloadInProgress。只要客戶端保持連接,服務(wù)器就從文件中讀取字節(jié)塊并發(fā)送給客戶端。對于多部分下載,這段代碼會發(fā)送特定的頭信息。如果客戶端中斷連接,服務(wù)器就把文件狀態(tài)設(shè)置為fsDownloadBroken。如果服務(wù)器完成了被請求范圍的發(fā)送過程,它會把狀態(tài)設(shè)置為fsDownloadFinished(見下載代碼)。

  FileInformation輔助類

  在ZIPHandler部分中你會發(fā)現(xiàn),F(xiàn)ileInformation是一個輔助類,它封裝了下載狀態(tài)信息(例如下載中、中斷等等)。

  為了建立FileInformation的實(shí)例,你需要把被請求文件的路徑傳遞給該類的構(gòu)造函數(shù):

Public Sub New(ByVal sPath As String)
 m_objFile = New System.IO.FileInfo(sPath)
End Sub

  FileInformation使用System.IO.FileInfo對象來獲取文件的信息,這些信息是作為該對象的屬性暴露的(例如文件是否存在、文件全名、大小等等)。這個類還暴露了一個DownloadState枚舉,它描述了下載請求的多種狀態(tài):

Enum DownloadState
 ' Clear:沒有下載過程,文件可能在維護(hù)
 fsClear = 1
 ' Locked:動態(tài)建立的文件不能被更改
 fsLocked = 2
 ' In Progress:文件被鎖定了,下載過程正在進(jìn)行
 fsDownloadInProgress = 6
 ' Broken:文件被鎖定了,下載過程正在進(jìn)行,但是被取消了
 fsDownloadBroken = 10
 ' Finished:文件被鎖定了,下載過程完成了
 fsDownloadFinished = 18
End Enum

  FileInformation還提供了EntityTag屬性值。示例代碼中的這個值是硬編碼的,這是由于示例代碼只使用了一個下載文件,并且該文件不會被改變,但是對于實(shí)際應(yīng)用程序來說,你會提供多個文件,甚至于動態(tài)地建立文件,你的代碼必須為每個文件提供一個唯一的EntityTag值。此外,每次改變或修改該文件的時候,這個值也必須改變。這使客戶端軟件能夠驗(yàn)證它們已經(jīng)下載的字節(jié)塊是否仍然是最新的。下面是示例代碼中返回硬編碼EntityTag值的部分:

Public ReadOnly Property EntityTag() As String
 ' EntityTag用于對客戶端的初始(200)響應(yīng),以及來自客戶端的恢復(fù)請求
 Get
  ' 為文件建立唯一的字符串。
  ' 注意,只要文件沒有發(fā)生改變,該唯一碼就必須保留。
  ' 但是,如果文件的確改變了或者被修改了,這個碼必須改變。
  Return "MyExampleFileID"
 End Get
End Property

  一個簡單的和大致足夠安全的EntityTag可能由文件名和文件最后被修改的日期組成。無論使用什么方法,你都必須確保這個值是真的是唯一的,不會與其它文件的EntityTag混淆。我希望在自己的應(yīng)用程序中按照客戶、顧客和郵編索引來動態(tài)地替被建立的文件命名,并把用作EntityTag的GUID存儲在數(shù)據(jù)庫中。

  ZipFileHandler類讀取和設(shè)置公共的State屬性。在完成下載以后,它把State設(shè)置為fsDownloadFinished。這個時候你就可以刪除臨時文件了。這兒一般需要調(diào)用Save方法來維持狀態(tài)。

Public Property State() As DownloadState
 Get
  Return m_nState
 End Get
 Set(ByVal nState As DownloadState)
  m_nState = nState
  ' 可選操作:這個時候你可以自動地刪除文件。
  ' 如果狀態(tài)被設(shè)置為Finished ,你就再也不需要這個文件了。
  ' If nState =DownloadState.fsDownloadFinished Then
   ' Clear()
  ' Else
   ' Save()
  ' End If
  Save()
 End Set
End Property

  在文件狀態(tài)發(fā)生改變的任何時候ZipFileHandler都應(yīng)該調(diào)用Save方法,保存文件的狀態(tài),這樣在以后才能顯示給用戶。你還可以用它來保存你自己建立的EntityTag。請不要把文件的狀態(tài)和EntityTag值保存在Application、Session或Cache中--你必須跨越所有的這些這些對象的生命周期來保存信息。

Private Sub Save()
 ' 把該文件下載的狀態(tài)保存到數(shù)據(jù)庫或XML文件中。
 ' 當(dāng)然,如果你并沒有動態(tài)地建立文件,就不需要保存這個狀態(tài)。
End Sub

  前面提到,示例代碼只處理一個已有的文件(download.zip),但是你可以進(jìn)一步增強(qiáng)這個程序,根據(jù)需要建立被請求的文件。

  測試示例代碼的時候,你的本地系統(tǒng)或LAN可能太快了,以至于無法中斷下載過程,因此我推薦你使用慢速LAN連接(在IIS中減少站點(diǎn)的帶寬是一種模擬的方法)或者把服務(wù)器放到互聯(lián)網(wǎng)上。

  在客戶端上下載文件仍然很艱難。ISP操作的不對的或配置錯誤的Web緩沖服務(wù)器都可能使大文件下載過程失敗,包括下載狀況惡化或早期對話終結(jié)。如果文件大小超過了255MB,你就應(yīng)該鼓勵顧客使用第三方下載管理軟件,盡管某些最新的瀏覽器內(nèi)建了基本的下載管理器。

    如果你希望進(jìn)一步擴(kuò)展示例代碼,查閱一下HTTP規(guī)范是有益的。你可以為下載建立MD5校驗(yàn)值,使用Content-MD5頭信息添加它們,提供一種驗(yàn)證下載文件完整性的途徑。示例代碼除了GET和HEAD之外沒有涉及到其它的HTTP方法。