什么是Http的斷點(diǎn)上傳和下載
斷點(diǎn)上傳:在向服務(wù)商上傳大文件的時(shí)候,將一個(gè)大的文件拆分成多個(gè)小的文件,每個(gè)文件通過(guò)單獨(dú)的Http請(qǐng)求上傳給服務(wù)器。
斷點(diǎn)下載:在向服務(wù)器請(qǐng)求下載一個(gè)大的資源文件的時(shí)候,不是一次Http請(qǐng)求返回所有的資源文件內(nèi)容。而是先通過(guò)Head請(qǐng)求,拿到資源文件的大小(單位:字節(jié))。然后每次請(qǐng)求只請(qǐng)求一部分字節(jié)的數(shù)據(jù),將請(qǐng)求到的數(shù)據(jù)在本地進(jìn)行拼接。
斷點(diǎn)上傳和下載的優(yōu)點(diǎn)
1、避免網(wǎng)絡(luò)中斷時(shí),重傳所有資源文件內(nèi)容。
2、提高服務(wù)器并發(fā),防止單個(gè)客戶端長(zhǎng)時(shí)間和服務(wù)器保持連接。
3、可以實(shí)時(shí)顯示上傳和下載的進(jìn)度。
斷點(diǎn)上傳和下載的缺點(diǎn)
1、占用更多的網(wǎng)絡(luò)帶寬,因?yàn)槊看蜨ttp請(qǐng)求都會(huì)附帶各種額外的信息。
2、上傳和下載的時(shí)間會(huì)變得長(zhǎng)一點(diǎn),因?yàn)槭峭ㄟ^(guò)多次請(qǐng)求來(lái)完成斷點(diǎn)上傳和下載。
實(shí)現(xiàn)基本原理
依賴Http協(xié)議的幾個(gè)基本的協(xié)議頭來(lái)完成斷點(diǎn)上傳和下載。
1、Content-Range
:這是一個(gè)響應(yīng)頭,表示請(qǐng)求的資源文件大小,我們可以通過(guò)Head請(qǐng)求拿到的資源文件的字節(jié)數(shù),就是讀取的這個(gè)字段。
2、Range
:這是一個(gè)請(qǐng)求頭,表示客戶端要請(qǐng)求的數(shù)組的范圍。如如:"0-1000"、"1001-2000"、"2001-3000"等,服務(wù)器接收到這個(gè)請(qǐng)求頭之后,只給我們返回對(duì)應(yīng)范圍內(nèi)的資源字節(jié)數(shù)組,不會(huì)把所有的字節(jié)數(shù)都返回給我們。
一般請(qǐng)求下,這兩個(gè)請(qǐng)求頭就可以實(shí)現(xiàn)簡(jiǎn)單的斷點(diǎn)上傳和下載。本篇文章我們使用一個(gè)WPF項(xiàng)目演示斷點(diǎn)下載。
string url = "http://file.cshelloworld.com/images/1771477326069108736.jpg";
long totalSize = 0;//文件總大小
long downLoadingSize = 0;//當(dāng)前已經(jīng)下載了多少
private void Button_Click(object sender, RoutedEventArgs e)
{
Task.Run(async () =>
{
//獲取到文件總大小 通過(guò)head請(qǐng)求
using HttpClient client = new HttpClient();
HttpRequestMessage requestMessage = new HttpRequestMessage(HttpMethod.Head, url);
var response = await client.SendAsync(requestMessage);
totalSize = response.Content.Headers.ContentLength.Value;
using FileStream fileStream = new FileStream("d:\\a.jpj", FileMode.Create, FileAccess.Write, FileShare.Read);
//開始分片下載
while (downLoadingSize < totalSize)
{
//組裝range 0,1000 1000,2000 0,9999
long start = downLoadingSize;
long end = start + 1000;
if (end > (totalSize - 1))
{
end = totalSize - 1;
}
client.DefaultRequestHeaders.Range = new System.Net.Http.Headers.RangeHeaderValue(start, end);
var res = await client.GetAsync(url);
byte[] bytes = await res.Content.ReadAsByteArrayAsync();
await fileStream.WriteAsync(bytes, 0, bytes.Length);
//更新UI的進(jìn)度
downLoadingSize += bytes.Length;
int process = (int)((downLoadingSize / (decimal)totalSize) * 100);
this.Dispatcher.Invoke(() =>
{
cont.Text = process + "%";
this.processBar.Value = process;
});
}
fileStream.Close();
});
}6
在以上代碼中,首先是Head請(qǐng)求獲取資源文件大小。
我們主要通過(guò)以下代碼實(shí)現(xiàn),通過(guò)設(shè)置HttpMethod.Head
構(gòu)建一個(gè)HttpRequestMessage
的請(qǐng)求對(duì)象
HttpRequestMessage requestMessage = new HttpRequestMessage(HttpMethod.Head, url);
其次是斷點(diǎn)下載過(guò)程中,Range請(qǐng)求頭如何設(shè)置:
client.DefaultRequestHeaders.Range = new System.Net.Http.Headers.RangeHeaderValue(start, end);
每次請(qǐng)求到字節(jié)數(shù)組之后,我們將字節(jié)數(shù)組寫入到本地的文件流中,如果網(wǎng)絡(luò)斷開,下次請(qǐng)求的時(shí)候,讀取本地文件大小,假設(shè)本地未見大小為1000,那么我們請(qǐng)求的時(shí)候Range就從1001開始,這樣服務(wù)器就給我們返回的是1001之后的字節(jié)數(shù)組了。
當(dāng)然在這個(gè)過(guò)程中,我們還要考慮一個(gè)問(wèn)題,如果服務(wù)器的資源文件發(fā)生了修改會(huì)怎么樣。如果我們繼續(xù)下載的話 ,就會(huì)出現(xiàn)問(wèn)題。因?yàn)榭蛻舳讼螺d的文件都不是同一個(gè)文件。這種情況下,我們可以使用Http的請(qǐng)求頭Last-Modified
來(lái)判斷文件是否修改,這個(gè)請(qǐng)求頭表示文件的最近一次修改時(shí)間。當(dāng)我們第一次請(qǐng)求數(shù)據(jù)的時(shí)候可以把這個(gè)請(qǐng)求頭的時(shí)間記錄下來(lái),后續(xù)請(qǐng)求如果服務(wù)器資源文件發(fā)生變化,我們就將本地文件全部刪除,然后重新發(fā)起請(qǐng)求。
閱讀原文:原文鏈接
該文章在 2025/3/31 11:04:12 編輯過(guò)