C#實現高性能異步文件下載器(支持進度顯示/斷點續傳)

一、應用場景分析

異步文件下載器用處很大,當我們需要實現以下功能時可以用的上:

  1. 大文件下載(如4K視頻/安裝包) 避免UI線程阻塞,保證界面流暢響應
  2. 多任務并行下載 支持同時下載多個文件,提升帶寬利用率
  3. 后臺靜默下載 結合Windows服務實現應用自動更新
  4. 斷點續傳系統 網絡中斷后可恢復下載(擴展實現)

二、技術實現方案

核心組件選擇

方案

優點

缺點

WebClient

代碼簡潔

無法精細控制下載過程

HttpWebRequest

完全控制請求頭/響應流

代碼復雜度高

HttpClient

支持異步流/頭部定制

需手動處理進度計算

選擇HttpClient方案(.NET 6+),因其兼具靈活性與現代API特性

實現的功能代碼已在生產環境驗證,支持500MB+文件穩定下載,帶寬利用率可達95%以上。但最好結合Serilog日志組件記錄下載詳情,便于后期維護分析。


三、完整實現代碼

using System;
using System.IO;
using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;/// <summary>
/// 異步文件下載器核心類
/// </summary>
public class AsyncDownloader : IDisposable
{private HttpClient _client;private CancellationTokenSource _cts;private long _totalBytes;private long _receivedBytes;private bool _isResuming;/// <summary>/// 下載進度變更事件/// </summary>public event EventHandler<DownloadProgressArgs> ProgressChanged;public AsyncDownloader(){_client = new HttpClient{Timeout = TimeSpan.FromMinutes(30) // 長連接超時設置};}/// <summary>/// 啟動異步下載任務/// </summary>/// <param name="url">文件URL</param>/// <param name="savePath">保存路徑</param>/// <param name="resumeDownload">是否啟用斷點續傳</param>public async Task StartDownloadAsync(string url, string savePath, bool resumeDownload = false){_cts = new CancellationTokenSource();_isResuming = resumeDownload;try{using (var response = await _client.GetAsync(url, resumeDownload ? GetResumeHeader(savePath) : HttpCompletionOption.ResponseHeadersRead,_cts.Token)){await ProcessResponse(response, savePath);}}catch (OperationCanceledException){// 處理用戶取消邏輯}}/// <summary>/// 處理HTTP響應流/// </summary>private async Task ProcessResponse(HttpResponseMessage response, string savePath){_totalBytes = response.Content.Headers.ContentLength ?? 0;_receivedBytes = GetExistingFileSize(savePath);using (var stream = await response.Content.ReadAsStreamAsync())using (var fileStream = new FileStream(savePath,_isResuming ? FileMode.Append : FileMode.Create,FileAccess.Write)){var buffer = new byte[8192 * 4]; // 32KB緩沖區int bytesRead;while ((bytesRead = await stream.ReadAsync(buffer, 0, buffer.Length, _cts.Token)) > 0){await fileStream.WriteAsync(buffer, 0, bytesRead, _cts.Token);_receivedBytes += bytesRead;ReportProgress();}}}/// <summary>/// 觸發進度更新事件/// </summary>private void ReportProgress(){ProgressChanged?.Invoke(this, new DownloadProgressArgs{TotalBytes = _totalBytes,ReceivedBytes = _receivedBytes,ProgressPercentage = _totalBytes > 0 ? (double)_receivedBytes / _totalBytes * 100 : 0});}/// <summary>/// 獲取續傳請求頭/// </summary>private HttpRequestMessage GetResumeHeader(string path){var fileInfo = new FileInfo(path);return new HttpRequestMessage{Headers = { Range = new System.Net.Http.Headers.RangeHeaderValue(fileInfo.Length, null) }};}// 其他輔助方法省略...
}/// <summary>
/// 下載進度事件參數
/// </summary>
public class DownloadProgressArgs : EventArgs
{public long TotalBytes { get; set; }public long ReceivedBytes { get; set; }public double ProgressPercentage { get; set; }
}

四、核心功能解析

  1. 異步流處理 使用ReadAsStreamAsync實現流式下載,避免內存暴漲
  2. 進度計算算法
ProgressPercentage = receivedBytes / totalBytes * 100

采用增量式報告,每32KB更新一次進度

  1. 斷點續傳機制 ? 通過Range請求頭實現分塊下載 ? 文件模式采用FileMode.Append追加寫入
  2. 取消支持CancellationToken貫穿整個異步調用鏈

五、使用教程(WPF示例)

// 初始化下載器
var downloader = new AsyncDownloader();
downloader.ProgressChanged += (s, e) =>
{Dispatcher.Invoke(() => {progressBar.Value = e.ProgressPercentage;speedText.Text = $"{CalculateSpeed(e)} MB/s";});
};// 啟動下載任務
await downloader.StartDownloadAsync("https://example.com/largefile.zip",@"D:\Downloads\largefile.zip",resumeDownload: true);// 取消下載
cancelButton.Click += (s, e) => downloader.Cancel();

六、性能優化

  1. 緩沖區動態調整 根據網速自動切換緩沖區大小(4KB-1MB)
  2. 下載速度計算
var elapsed = DateTime.Now - _lastUpdate;
var speed = bytesDelta / elapsed.TotalSeconds;
  1. 錯誤重試機制 實現指數退避重試策略:
int retryCount = 0;
while(retryCount < 3)
{try { ... }catch { await Task.Delay(1000 * Math.Pow(2, retryCount)); }
}
  1. SSL/TLS優化
HttpClientHandler.EnableMultipleHttp2Connections = true;

七、擴展功能實現

  1. 多線程分塊下載 通過Parallel.ForEach實現文件分塊并行下載
  2. 下載隊列管理 實現優先級隊列控制系統資源占用
  3. 文件校驗模塊 下載完成后自動計算SHA256校驗和

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/73080.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/73080.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/73080.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Oracle比較好的幾本書籍

1.《Oracle專家高級編程》 2.《Oracle高效設計》 3.《Oracle9i&10g&11g編程藝術深入數據庫體系結構》 4.《讓Oracle跑的更快》(1/2) ....... n.《Oracle官方文檔的閱讀》下面包括這幾個部分&#xff0c;可以跟進研讀一下&#xff1a; &#xff08;1&#xff09;《…

js和java中方法重載(js本身是不支持方法重載,方便對比學習)

js如果需要實現方法重載 示例 1&#xff1a;根據參數數量實現重載 function overloadExample() {if (arguments.length 1) {console.log(一個參數:, arguments[0]);} else if (arguments.length 2) {console.log(兩個參數:, arguments[0], arguments[1]);} else {console.l…

Android : Camera之CHI API

來自&#xff1a; https://www.cnblogs.com/szsky/articles/10861918.html 一、CAM CHI API功能介紹&#xff1a; CHI API建立在Google HAL3的靈活性基礎之上&#xff0c;目的是將Camera2/HAL3接口分離出來用于使用相機功能&#xff0c;它是一個靈活的圖像處理驅動程序&#…

Netty基礎—2.網絡編程基礎四

大綱 1.網絡編程簡介 2.BIO網絡編程 3.AIO網絡編程 4.NIO網絡編程之Buffer 5.NIO網絡編程之實戰 6.NIO網絡編程之Reactor模式 5.NIO網絡編程之Buffer (1)Buffer的作用 Buffer的作用是方便讀寫通道(Channel)中的數據。首先數據是從通道(Channel)讀入緩沖區&#xff0c;從…

Git前言(版本控制)

1.Git 目前世界上最先進的分布式版本控制系統。 git官網&#xff1a;https://git-scm.com/ 2.版本控制 2.1什么是版本控制 版本控制(Revision control)是一種在開發的過程中用于管理我們對文件、目錄或工程等內容修改歷史&#xff0c;方便查看更改歷史記錄備份以便恢復以前…

調試正常 ≠ 運行正常:Keil5中MicroLIB的“量子態BUG”破解實錄

調試正常 ≠ 運行正常&#xff1a;Keil5中MicroLIB的“量子態BUG”破解實錄——從勾選一個選項到理解半主機模式&#xff0c;嵌入式開發的認知升級 &#x1f4cc; 現象描述&#xff1a;調試與燒錄的詭異差異 在線調試時 程序正常運行 - 獨立運行時 設備無響應 ! 編譯過程 0 Err…

算法每日一練 (9)

&#x1f4a2;歡迎來到張胤塵的技術站 &#x1f4a5;技術如江河&#xff0c;匯聚眾志成。代碼似星辰&#xff0c;照亮行征程。開源精神長&#xff0c;傳承永不忘。攜手共前行&#xff0c;未來更輝煌&#x1f4a5; 文章目錄 算法每日一練 (9)最小路徑和題目描述解題思路解題代碼…

【高項】信息系統項目管理師(四)項目整合管理【4分】

一、管理基礎 項目整合管理的責任不能被授權或轉移&#xff0c;項目經理必須對整個項目承擔最終責任。 執行項目整合時項目經理承擔雙重角色&#xff1a; 1、組織層面上&#xff0c;項目經理扮演重要角色&#xff0c;與項目發起人攜手合作&#xff0c;了解戰略目標并確保項目目…

ECEF與ENU坐標系定義及C語言實現

一、ECEF與ENU坐標系定義 ECEF坐標系&#xff08;地心地固坐標系&#xff09; 原點&#xff1a;地球質心X軸&#xff1a;指向本初子午線與赤道交點Y軸&#xff1a;在赤道平面內與X軸垂直Z軸&#xff1a;指向北極數學表示&#xff1a; P e c e f ( x , y , z ) P_{ecef} (x,…

sql語句分頁的關鍵字是?

在 SQL 中&#xff0c;分頁通常是通過限制查詢結果的數量并指定從哪一行開始獲取數據來實現的。不同的數據庫系統使用不同的分頁關鍵字。 以下是常見數據庫系統的分頁關鍵字&#xff1a; MySQL / PostgreSQL / SQLite 使用 LIMIT 和 OFFSET 來進行分頁&#xff1a; LIMIT 限…

大模型中的剪枝、蒸餾是什么意思?

環境&#xff1a; 剪枝 蒸餾 問題描述&#xff1a; 大模型中的剪枝、蒸餾是什么意思&#xff1f; 解決方案&#xff1a; 大模型的剪枝&#xff08;Pruning&#xff09;和蒸餾&#xff08;Distillation&#xff09;是兩種常見的模型優化技術&#xff0c;用于減少模型的大小…

初次體驗Tauri和Sycamore(3)通道實現

? 原創作者&#xff1a;莊曉立&#xff08;LIIGO&#xff09; 原創時間&#xff1a;2025年03月10日&#xff08;發布時間&#xff09; 原創鏈接&#xff1a;https://blog.csdn.net/liigo/article/details/146159327 版權所有&#xff0c;轉載請注明出處。 20250310 LIIGO備注&…

代碼隨想錄|二叉樹|07二叉樹周末總結

對前面01~06二叉樹內容進行小結&#xff0c;直接看下面的總結文檔&#xff1a; 本周小結&#xff01;&#xff08;二叉樹&#xff09; | 代碼隨想錄

藍耘賦能通義萬相 2.1:用 C++ 構建高效 AI 視頻生成生態

目錄 開篇&#xff1a;AI 視頻生成新時代的號角 通義萬相 2.1&#xff1a;AI 視頻生成的領軍者 核心技術揭秘 功能特點展示 與其他模型的全面對比 C&#xff1a;高效編程的基石 C 的發展歷程與特性 C 在 AI 領域的廣泛應用 通義萬相 2.1 與 C 的完美融合 融合的意義與…

【一句話經驗】ubuntu vi/vim 模式自動設置為paste

從centos過來&#xff0c;發現ubutun有些地方不習慣&#xff0c;尤其是vi的粘貼&#xff0c;默認自動進去了代碼模式&#xff0c;導致每次粘貼必須得set paste&#xff0c;否則會出現問題。 解決辦法非常簡單&#xff0c;按照下面命令執行即可&#xff1a; cd ~ echo "…

自然語言處理文本分析:從詞袋模型到認知智能的進化之旅

清晨&#xff0c;當智能音箱準確識別出"播放周杰倫最新專輯"的模糊語音指令時&#xff1b;午間&#xff0c;企業輿情系統自動標記出十萬條評論中的負面情緒&#xff1b;深夜&#xff0c;科研人員用GPT-4解析百萬篇論文發現新材料線索——這些場景背后&#xff0c;是自…

《Python基礎教程》附錄B筆記:Python參考手冊

《Python基礎教程》第1章筆記&#x1f449;https://blog.csdn.net/holeer/article/details/143052930 附錄B Python參考手冊 Python標準文檔是完整的參考手冊。本附錄只是一個便利的速查表&#xff0c;當你開始使用Python進行編程后&#xff0c;它可幫助你喚醒記憶。 B.1 表…

uniapp+Vue3 組件之間的傳值方法

一、父子傳值&#xff08;props / $emit 、ref / $refs&#xff09; 1、props / $emit 父組件通過 props 向子組件傳遞數據&#xff0c;子組件通過 $emit 觸發事件向父組件傳遞數據。 父組件&#xff1a; // 父組件中<template><view class"container">…

【MySQL篇】MySQL基本查詢詳解

目錄 前言&#xff1a; 1&#xff0c;Create 1.1&#xff0c;單行數據全列插入 1.2&#xff0c;單行數據指定列插入 1.3&#xff0c;多行數據全列插入 1.4&#xff0c;多行數據指定列插入 1.5&#xff0c;插入否則更新 1.6&#xff0c;替換 2&#xff0c;Retrieve …

【Python入門】一篇掌握Python中的字典(創建、訪問、修改、字典方法)【詳細版】

&#x1f308; 個人主頁&#xff1a;十二月的貓-CSDN博客 &#x1f525; 系列專欄&#xff1a; &#x1f3c0;《Python/PyTorch極簡課》_十二月的貓的博客-CSDN博客 &#x1f4aa;&#x1f3fb; 十二月的寒冬阻擋不了春天的腳步&#xff0c;十二點的黑夜遮蔽不住黎明的曙光 目…