C#中WebClient實(shí)現(xiàn)文件下載
鑒于各種復(fù)雜的網(wǎng)絡(luò)環(huán)境,筆者決定采用不同的編程接口進(jìn)行下載嘗試,以增加程序的可用性。
這里僅介紹使用 WebClient 的方法,后續(xù)的文章會(huì)介紹其他的方法。博文中主要介紹思路和關(guān)鍵代碼,完整的 demo 附在文末。
使用代理訪問(wèn)網(wǎng)絡(luò)
很多公司的員工都是通過(guò)公司設(shè)置的代理上網(wǎng)的。通過(guò)代理上網(wǎng)主要是方便公司進(jìn)行各種的管制,當(dāng)然也能實(shí)現(xiàn)一些特殊的功能… 不過(guò)這會(huì)給我們的程序訪問(wèn)網(wǎng)絡(luò)帶來(lái)一些問(wèn)題。
其實(shí),WebClient 中的 API 已經(jīng)很智能了,比如我們創(chuàng)建的 HttpWebRequest 對(duì)象,它自帶一個(gè) Proxy 屬性。也就是說(shuō),WebHttpRequest 默認(rèn)會(huì)使用找到的代理。這很棒,也能處理很多情況了。可是如果這個(gè)默認(rèn)的代理需要驗(yàn)證域用戶的身份信息,這時(shí)使用 WebHttpRequest 訪問(wèn)網(wǎng)絡(luò)就可能失敗。此時(shí)查看 Proxy. Credentials 屬性,發(fā)現(xiàn)它是 null。
從 WebClient 的 API 中是可以取到系統(tǒng)默認(rèn)的 Credentials 的,只是不太清楚為什么 Proxy.Credentials 屬性默認(rèn)沒有設(shè)置為這個(gè)值。我們自己設(shè)置下就可以了。
request.Proxy.Credentials = CredentialCache.DefaultCredentials;
但實(shí)際的網(wǎng)絡(luò)環(huán)境可能會(huì)更復(fù)雜,需要用戶來(lái)指定聯(lián)網(wǎng)的代理,并同時(shí)指定聯(lián)網(wǎng)所需的 Credentials。寫法如下:
myProxy = new WebProxy("proxyAddress"); myProxy.Credentials = new NetworkCredential(ProxyUserName, ProxyUserPasswd, DomainName);
克服緩存
緩存可謂無(wú)處不再,在服務(wù)器端 CDN 會(huì)有緩存,在客戶端的代理層也會(huì)有緩存。所以經(jīng)常出現(xiàn)的問(wèn)題是:服務(wù)器上的文件明明更新了,還是會(huì)有一些客戶下載到舊文件。我們先來(lái)處理客戶端的緩存問(wèn)題。
HttpWebRequest 的 CachePolicy.Level 屬性就是設(shè)置緩存策略的,只是它的默認(rèn)值是 BypassCache。我們把它改為 Reload 就行了:
request.CachePolicy = new System.Net.Cache.RequestCachePolicy(System.Net.Cache.RequestCacheLevel.Reload);
接下來(lái)是服務(wù)器端的緩存問(wèn)題。
現(xiàn)在大家好像都在使用 CDN,可在使用中經(jīng)常發(fā)現(xiàn) CDN 端的緩存更新有問(wèn)題。在網(wǎng)上查了查也沒有什么好的解決辦法,不過(guò)倒是有一個(gè)很好的 workaround,就是在請(qǐng)求中添加一個(gè)隨機(jī)的字符串作為參數(shù)。
Random rdm = new Random(); string s = rdm.Next().ToString(); myUrl += "?" + s;
需要注意的是,關(guān)于緩存,一定要使用符合當(dāng)前用例的策略,且不可搞一刀切。
更友好的下載過(guò)程
使用滾動(dòng)條顯示下載進(jìn)度,顯示實(shí)時(shí)的下載速度,允許用戶取消下載:
下面是下載用的核心代碼,我們把它分為計(jì)算下載百分比和計(jì)算當(dāng)前下載速度分別介紹。
// 獲得下載文件的長(zhǎng)度 double contentLength = DownloadManager.GetContentLength(myHttpWebClient); byte[] buffer = new byte[BufferSize]; long downloadedLength = 0; long currentTimeSpanDataLength = 0; int currentDataLength; while ((currentDataLength = stream.Read(buffer, 0, BufferSize)) > 0 && !this._cancelDownload) { fileStream.Write(buffer, 0, currentDataLength); downloadedLength += (long)currentDataLength; currentTimeSpanDataLength += (long)currentDataLength; int intDownloadSpeed = 0; if (this._downloadStopWatch.ElapsedMilliseconds > 800) { double num5 = (double)currentTimeSpanDataLength / 1024.0; double num6 = (double)this._downloadStopWatch.ElapsedMilliseconds / 1000.0; double doubleDownloadSpeed = num5 / num6; intDownloadSpeed = (int)Math.Round(doubleDownloadSpeed, 0); this._downloadStopWatch.Reset(); this._downloadStopWatch.Start(); currentTimeSpanDataLength = 0; } double doubleDownloadPersent = 0.0; if (contentLength > 0.0) { doubleDownloadPersent = (double)downloadedLength / contentLength; } }
在下載的過(guò)程中計(jì)算下載百分比
首先需要從 http 請(qǐng)求中獲得要下載文件的長(zhǎng)度,細(xì)節(jié)請(qǐng)參考本文所配 demo。
double contentLength = DownloadManager.GetContentLength(myHttpWebClient);
每從文件流中讀取一次數(shù)據(jù),我們知道讀了多少個(gè)字節(jié)(currentDataLength),累計(jì)下來(lái)就是當(dāng)前已經(jīng)下載了的文件長(zhǎng)度。
downloadedLength += (long)currentDataLength;
然后做個(gè)除法就行了:
doubleDownloadPersent = (double)downloadedLength / contentLength;
計(jì)算實(shí)時(shí)的下載速度
對(duì)于當(dāng)前的下載速度,我們計(jì)算過(guò)去的一段時(shí)間內(nèi)下載下來(lái)的字節(jié)數(shù)。時(shí)間段可以使用 StopWatch 來(lái)獲得,我選擇的時(shí)間段要求大于 800 毫秒。
if (this._downloadStopWatch.ElapsedMilliseconds > 800) { /***********************************/ // 計(jì)算上一個(gè)時(shí)間段內(nèi)的下載速度 double num5 = (double)currentTimeSpanDataLength / 1024.0; double num6 = (double)this._downloadStopWatch.ElapsedMilliseconds / 1000.0; double doubleDownloadSpeed = num5 / num6; /***********************************/ intDownloadSpeed = (int)Math.Round(doubleDownloadSpeed, 0); // 本次網(wǎng)速計(jì)算完成后重置時(shí)間計(jì)時(shí)器和數(shù)據(jù)計(jì)數(shù)器,開始下次的計(jì)算 this._downloadStopWatch.Reset(); this._downloadStopWatch.Start(); currentTimeSpanDataLength = 0; }
事實(shí)上每次計(jì)算下載速度的時(shí)間段長(zhǎng)度是不顧定的,但這并不影響計(jì)算結(jié)果,我只要保證距離上次計(jì)算超過(guò)了 800 毫秒就行了。
允許用戶取消下載
對(duì)于一個(gè)執(zhí)行時(shí)間比較長(zhǎng)的任務(wù)來(lái)說(shuō),不允許用戶取消它是被深惡痛絕的!尤其是網(wǎng)速不太好的時(shí)候。所以我們需要給用戶一個(gè)選擇:可以痛快(而不是痛苦)的結(jié)束當(dāng)前的旅程。
而這一切對(duì)我們來(lái)說(shuō)又是那么的簡(jiǎn)單!
while ((currentDataLength = stream.Read(buffer, 0, BufferSize)) > 0 && !this._cancelDownload){}
當(dāng)從數(shù)據(jù)流中讀取數(shù)據(jù)時(shí),我們檢查用戶是不是按下了"取消"按鈕,就是這里的 this._cancelDownload 變量。如果它是 true 就結(jié)束當(dāng)前的下載。
至此,把用戶抱怨最多的幾個(gè)點(diǎn)都搞定了。其實(shí)也沒有增加多少代碼,并且每個(gè)知識(shí)點(diǎn)看起來(lái)都是那么的細(xì)微。但很明顯的提高了用戶的使用體驗(yàn)。這也給我們帶來(lái)了一些啟發(fā),完成主要功能可能只是工作中的一部分,另外的一些工作可能并不是那么明顯,需要我們不斷的體會(huì),發(fā)覺…
Demo 下載地址:WebClientDemo_jb51.rar
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C# 站點(diǎn)IP訪問(wèn)頻率限制 針對(duì)單個(gè)站點(diǎn)的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇C# 站點(diǎn)IP訪問(wèn)頻率限制 針對(duì)單個(gè)站點(diǎn)的實(shí)現(xiàn)方法。小編覺的挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12C#程序提示“正由另一進(jìn)程使用,因此該進(jìn)程無(wú)法訪問(wèn)該文件”的解決辦法
這篇文章主要介紹了C#程序提示“正由另一進(jìn)程使用,因此該進(jìn)程無(wú)法訪問(wèn)該文件”的解決辦法,本文通過(guò)改寫程序代碼實(shí)現(xiàn)解決這個(gè)問(wèn)題,需要的朋友可以參考下2015-06-06詳解WPF如何動(dòng)態(tài)生成DataGrid的行和列
在日常開發(fā)中,DataGrid作為二維表格,非常適合數(shù)據(jù)的展示和統(tǒng)計(jì),本文以一些簡(jiǎn)單的小例子,簡(jiǎn)述在WPF開發(fā)中,如何動(dòng)態(tài)生成DataGrid的行和列,需要的可以了解下2024-02-02C#實(shí)現(xiàn)PDF合并的項(xiàng)目實(shí)踐
有時(shí)我們可能會(huì)遇到需要的資料或教程被分成了幾部分存放在多個(gè)PDF文件中,本文主要介紹了C#實(shí)現(xiàn)PDF合并的項(xiàng)目實(shí)踐,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01C#.NET中如何批量插入大量數(shù)據(jù)到數(shù)據(jù)庫(kù)中
這篇文章主要給大家介紹C#.net中如何批量插入大量數(shù)據(jù)到數(shù)據(jù)庫(kù)中,本文涉及到C#.net中批量插入數(shù)據(jù)到數(shù)據(jù)庫(kù)中方面的內(nèi)容,對(duì)C#.net批量插入數(shù)據(jù)到數(shù)據(jù)庫(kù)中感興趣的朋友可以參考下本篇文章2015-10-10C#使用GZipStream實(shí)現(xiàn)文件的壓縮與解壓
這篇文章主要為大家詳細(xì)介紹了C#使用GZipStream實(shí)現(xiàn)文件的壓縮與解壓,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-10-10