Flutter 下載篇 - 叄 | 網路庫切換實踐與思考

来源:https://www.cnblogs.com/xuge2it/archive/2023/03/04/17178595.html
-Advertisement-
Play Games

前言 本文是關於使用flutter_download_manager下載功能的實踐和探索。我們將基於flutter_download_manager的功能擴展,改造成自己想要的樣子。在閱讀本文之前,建議先瞭解前兩篇文章: Flutter 下載篇 - 壹 | flutter_download_mana ...


前言

本文是關於使用flutter_download_manager下載功能的實踐和探索。我們將基於flutter_download_manager的功能擴展,改造成自己想要的樣子。在閱讀本文之前,建議先瞭解前兩篇文章:

本文將基於第二篇中的擴展框架,將網路庫從dio切換為httpclient,並結合改造過程中發現的問題提出自己的想法。

優化點:dynamic的告警問題

Untitled.png

在第2和20行中,黃色標記表明,如果第2行中的每個網路庫下載的返回值可能不同,則考慮將其設置為“dynamic”,這可能導致第20行中出現響應狀態碼的告警,因為該屬性可能不存在。

為了確保 download 介面的返回值具有 statusCode 屬性,在這裡定義了一個專門的返回類以進行限制。具體定義如下:

Untitled 1.png

這樣就解決了statusCode告警問題,其中extra可以存放原始download response對象。

HttpClient實現網路庫

只用實現上一篇中的網路介面CustomHttpClient,然後修改IDownloader:createObject其中網路註入對象即可,如下:

//實現這個介面定義
abstract class CustomHttpClient {
  Future<DownloadHttpResponse> download(
    String urlPath,
    String savePath, {
    DownloadProgressCallback? onReceiveProgress,
    DownloadCancelToken? cancelToken,
    Map<String, dynamic>? options,
  });

  DownloadCancelToken generateToken();
}

------【idownloader.dart】----------
abstract class IDownloader {
  factory IDownloader() => createObject(
        //將這個註入修改成我們實現的即可 原來:customHttpClient: CustomDioImpl(),
        customHttpClient: CustomHttpClientImp(),
      );
}

實現代碼:

Untitled 2.png

  • 第9-17行:主要是將flutter_download_manager中已下載但未下載完整的文件大小傳遞給後端,以便告知後端從哪裡繼續下載文件。

如果不傳,會浪費帶寬和時間。在處理大文件時,記憶體壓力會增大,中斷的可能性也會增加。此外,用戶界面可能會出現進度條跳躍的問題。

  • 第27-45行:將下載流寫入傳入的 savepath 文件中。需要註意 cancelToken.isCancelled 方法,因為上一篇中沒有定義 isCancelled 屬性,這裡必須在 DownloadCancelToken 中提供該方法(第69行)。
  • 第55-65行:這裡實現了HttpClientCancelToken的cancel方法,具體實現就是給標誌位_isCancelled賦值。

遇到官方問題

完成上述實踐後,發現官方進度錯誤BUG。如果多次暫停、取消,然後再恢復下載,會出現進度起始位置錯誤的問題。下載會從已下載文件的長度開始,效果如下所示:

221539959-e5af41bc-b3b1-41cc-9a46-1ba549c4fd86.gif

問題原因

在暫停時,暫停前未將下載流寫入已下載的文件中。

解決辦法

如果用戶點擊了暫停,會拋出取消異常,此時捕獲該異常並判斷當前下載任務狀態是暫停態,將已下載的數據流寫入未下載完全的文件中。

已提交PR到官方庫中,見PR地址

完整代碼傳送門,其中包含了httpclient實現和上述官方進度問題修複方案。

回顧網路庫解耦

在切換flutter_download_manager網路庫時,我們發現解耦方案仍然存在問題。

1. isCanceled

在httpclient中使用了isCancelled方法,不得不將其加入DownloadCancelToken中,這在設計上是有問題的。

我查看了dio的download過程,發現其中也存在對取消狀態的判斷。dio.CancelToken類中也定義了這個方法,那麼為什麼我沒有考慮到呢?原因是我沒有實踐過,當時只是用downloadTokenProxy去代理了CancelToken,它可以跑,就認為設計沒有問題。果然,自己挖的坑需要自己踩一遍才能真正理解其中的問題。

2. flutter_download_manager框架運行約束

為了讓該庫正常運行,必須與相關的網路庫配合使用。

在我使用httpclient進行實現過程中,我發現如果取消操作,必須拋出一個異常(請參考代碼中第32行),才能確保程式能夠順利地執行case1而不出現官方文檔中提到的問題。

Future<void> download(
      String url, String savePath, DownloadCancelToken cancelToken,
      {forceDownload = false}) async {
    late String partialFilePath;
    late File partialFile;
    try {
      var task = getDownload(url);
        var response = await customHttpClient.download(...);
      } else {
        var response = await customHttpClient.download(...
        );
      }
    } catch (e) {
      var task = getDownload(url)!;
      if (task.status.value != DownloadStatus.canceled &&
        //...
      } else if (task.status.value == DownloadStatus.paused) {
        // 只有拋出取消異常才能走到保持下載流到未下載完全文件中 case1
        final ioSink = partialFile.openWrite(mode: FileMode.writeOnlyAppend);
        final f = File(partialFilePath + tempExtension);
        final fileExist = await f.exists();
        if (fileExist) {
          await ioSink.addStream(f.openRead());
          await f.delete();
        }
        await ioSink.close();
      }
    }

約束一:如果需要取消任務,應該拋出取消異常。

因為flutter_download_manager一開始網路庫就是綁定的dio,而dio中對取消操作的結果反饋就是取消異常。如果用戶取消了任何一個請求,就會拋出該異常。

話說,取消發送一條消息難道非得拋出異常才可以嗎?其實有很多方法可以實現這個功能。

約束二:請提供下載請求的返回碼。

由於flutter_download_manager已經處理了返回碼206和200,如果不提供網路請求返回碼,相關邏輯無法執行。

話說,請求成功返回結果的方式也可以是發消息吧。

下載框架設計思路

如果將flutter_download_manager作為代碼片段使用是沒有問題的,但從下載框架設計的角度來看,仍需要進一步改進和優化。

出現上述提到的約束問題,主要是將關係集中在DownloadManager和網路庫上,陷入網路細節中。實際上,這兩者沒有直接關係,主要是flutter_download_manager作者將它們耦合在一起導致的。

從下載框架角度說,類之間依賴關係應該如下:

Untitled 3.png

DownloadManager依賴下載器,下載器依賴網路庫

三者間交互關係如下:

Untitled 4.png

  • DownloadManager 通過維護列表來管理內部任務的增刪改查。每個任務對應一個下載過程。
  • Downloader 負責任務下載,並通過同步或非同步消息通知當前下載任務的狀態。DownloadManger 通過這些消息來更新任務列表。
  • Downloader 通過向網路庫發送請求來下載任務。網路將結果返回給 Downloader,由 Downloader 來決定內部狀態和斷點續傳邏輯。

總結

本文介紹了Flutter下載功能的實踐和探索,包括網路庫的切換和優化。使用了httpclient實現網路庫,並解決了官方進度錯誤BUG。還回顧了flutter_download_manager的設計缺陷,並提出了下載框架的設計思路。總之,提供了有關Flutter下載功能的實用信息和思考。

太棒了!鼓勵自己堅持到底。我希望我為你投入的時間增加了一些價值。

如果覺得文章對你有幫助,點贊、收藏、關註、評論,一鍵四連支持,你的支持就是我創作最大的動力。

❤️本文由公眾號編程黑板報 原創,關註我,獲取我的最新文章~❤️


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 背景 當我們在用Avalonia開發項目的時候會涉及到路徑的問題,在Linux下麵路徑的寫法和Window路徑的寫法有區別 問題描述 用Avalonia開發項目發現我們之前的路徑寫法在Linux下麵運行報錯,尤其是在結合Prism開發寫項目進行模塊化掃描的方式載入我們的模塊程式就會出現報錯 解決方案 ...
  • 頁眉位於文檔中每個頁面的頂部區域,常用於顯示文檔的附加信息,可以插入時間、圖形、公司微標、文檔標題、文件名或作者姓名等;頁腳位於文檔中每個頁面的底部的區域,常用於顯示文檔的附加信息,可以在頁腳中插入文本或圖形。今天這篇文章就將為大家展示如何以編程的方式在在 Word 文檔中添加頁眉和頁腳。下麵是我整 ...
  • 實現背景 最近需要把大數據測試環境的虛擬機遷移到另一臺物理機上,其中資料庫文件過大,一般的上傳文件有大小限制不能超過4GB,可以通過NFS共用目錄解決文件上傳問題。 NFS介紹 NFS介紹NFS 即網路文件系統(Network File-System),可以通過網路讓不同機器、不同系統之間可以實現文 ...
  • 原文:What is a Shell? 註意:本文是對原文的一個翻譯! Shell是一個巨集處理器,用於執行命令。巨集處理器意味著,將文本和符號展開,創建一個更大的表達式。 Unix shell 既是一個命令解釋器,也是一門編程語言。作為一個命令解釋器,它提供了豐富的GUN工具集可以與用戶進行交互。作為 ...
  • 前言 本文展示了一個比較完整的企業項目級別的Makefile文件,包括了:文件調用,源文件、頭文件、庫文件指定,軟體版本號、巨集定義,編譯時間,自動目錄等內容。 1、目錄架構 本文中所採用的目錄架構,在企業項目開發中十分常見:源文件都放在src目錄中,頭文件都放在inc目錄中,並且這兩個目錄都可以有對 ...
  • 1602LCD 是工業上常用的模塊, 在工廠交通運輸設備上經常能見到. 驅動晶元為 HD44780, 1602LCD 的字元顯示為兩行, 每行16個字元, 字元基於5×8的像素矩陣 ...
  • 1、工作應用場景 統計得到每個小時的UV、PV、IP的個數,構建如下表結構: 但是表中數據的存儲格式不利於直接查詢展示,需要進行調整:(以時間分區,去重、聚合等……對結果進行行列轉換) 2、行轉列 (1)多行轉多列 case when函數 功能:用於實現對數據的判斷,根據條件,不同的情況返回不同的結 ...
  • 1、全局排序(Order by) 功能:全局排序,只有1個reducer(用1個Reduce Task完成全局排序,與設置的Reduce Task個數無關) 參數:ASC:升序(預設) DESC:降序 使用:order by放在select語句的結尾 例如: --查詢員工信息按工資降序排列 sele ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...