C# 斷點續傳原理與實現

iDotNetSpace發表於2009-01-14

在瞭解HTTP斷點續傳的原理之前,讓我們先來了解一下HTTP協議,HTTP協議是一種基於tcp的簡單協議,分為請求和回覆兩種。請求協議是由客戶機(瀏覽器)向伺服器(WEB SERVER)提交請求時傳送報文的協議。回覆協議是由伺服器(web server),向客戶機(瀏覽器)回覆報文時的協議。請求和回覆協議都由頭和體組成。頭和體之間以一行空行為分隔。

   以下是一個請求報文與相應的回覆報文的例子:
GET /image/index_r4_c1.jpg HTTP/1.1
Accept: */*
Referer: http://192.168.3.120:8080
Accept-Language: zh-cn
Accept-Encoding: gzip, deflate
User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.0; .NET CLR 1.0.3705)
Host: 192.168.3.120:8080
Connection: Keep-Alive

HTTP/1.1 200 OK
Server: Microsoft-IIS/5.0
Date: Tue, 24 Jun 2003 05:39:40 GMT
Content-Type: image/jpeg
Accept-Ranges: bytes
Last-Modified: Thu, 23 May 2002 03:05:40 GMT
ETag: "bec48eb862c21:934"
Content-Length: 2827

….

  下面我們就來說說"斷點續傳",顧名思義,斷點續傳就是在上一次下載時斷開的位置開始繼續下載。
在HTTP協議中,可以在請求報文頭中加入Range段,來表示客戶機希望從何處繼續下載。  

  比如說從第1024位元組開始下載,請求報文如下: 

GET /image/index_r4_c1.jpg HTTP/1.1
Accept: */*
Referer: http://192.168.3.120:8080
Accept-Language: zh-cn
Accept-Encoding: gzip, deflate
User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.0; .NET CLR 1.0.3705)
Host: 192.168.3.120:8080
Range:bytes=1024-
Connection: Keep-Alive
 

  .NET中的相關類

   明白了上面的原理,那麼,我們來看看.NET FRAMEWORK中為我們提供了哪些類可以來做這些事。

   完成HTTP請求

System.Net.HttpWebRequest

   HttpWebRequest 類對 WebRequest 中定義的屬性和方法提供支援,也對使使用者能夠直接與使用 HTTP 的伺服器互動的附加屬性和方法提供支援。

   HttpWebRequest 將傳送到 Internet 資源的公共 HTTP 標頭值公開為屬性,由方法或系統設定。下表包含完整列表。可以將 Headers 屬性中的其他標頭設定為名稱/值對。但是注意,某些公共標頭被視為受限制的,它們或者直接由 API公開,或者受到系統保護,不能被更改。Range也屬於被保護之列,不過,.NET為開發者提供了更方便的操作,就是 AddRange方法,向請求新增從請求資料的開始處或結束處的特定範圍的位元組範圍標頭

   完成檔案訪問

System.IO.FileStream

   FileStream 物件支援使用Seek方法對檔案進行隨機訪問, Seek 允許將讀取/寫入位置移動到檔案中的任意位置。這是通過位元組偏移參考點引數完成的。位元組偏移量是相對於查詢參考點而言的,該參考點可以是基礎檔案的開始、當前位置或結尾,分別由SeekOrigin類的三個屬性表示。

   程式碼實現

   瞭解了.NET提供的相關的類,那麼,我們就可以方便的實現了。

   程式碼如下:
 

static void Main(string[] args)
{

string StrFileName="c:\\aa.zip"; //根據實際情況設定
string StrUrl="http://www.xxxx.cn/xxxxx.zip"; //根據實際情況設定

//開啟上次下載的檔案或新建檔案
long lStartPos =0;
System.IO.FileStream fs;
if (System.IO.File.Exists(StrFileName))
{
fs= System.IO.File.OpenWrite(StrFileName);
lStartPos=fs.Length;
fs.Seek(lStartPos,System.IO.SeekOrigin.Current); //移動檔案流中的當前指標
}
else
{
fs = new System.IO.FileStream(StrFileName,System.IO.FileMode.Create);
lStartPos =0;
}

//開啟網路連線
try
{
System.Net.HttpWebRequest request =(System.Net.HttpWebRequest)System.Net.HttpWebRequest.Create(StrUrl);
if ( lStartPos>0)
request.AddRange((int)lStartPos); //設定Range值

//向伺服器請求,獲得伺服器回應資料流
System.IO.Stream ns= request.GetResponse().GetResponseStream();

byte[] nbytes = new byte[512];
int nReadSize=0;
nReadSize=ns.Read(nbytes,0,512);
while( nReadSize >0)
{
fs.Write(nbytes,0,nReadSize);
nReadSize=ns.Read(nbytes,0,512);
}
fs.Close();
ns.Close();
Console.WriteLine("下載完成");
}
catch(Exception ex)
{
fs.Close();
Console.WriteLine("下載過程中出現錯誤:"+ex.ToString());
}
}

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/12639172/viewspace-539618/,如需轉載,請註明出處,否則將追究法律責任。

相關文章