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
  • .NET Core 選項系統的主要實現在 Microsoft.Extensions.Options 和 Microsoft.Extensions.Options.ConfigurationExtensions 兩個 Nuget 包。對於一個框架的源碼進行解讀,我們可以從我們常用的框架中的類或方法入手 ...
  • 最近在工作中遇到一個問題,就是我有多個線程會調用bitmap對象,運行的時候報錯,對象當前正在其他地方使用。第一反應肯定是加鎖啊,於是我就在每個用到bitmap的地方都加了鎖,但是運行之後依然報這個錯 測試代碼如下 using System; using System.Drawing; using ...
  • 一:背景 1. 講故事 前段時間有位朋友微信找到我,說他的程式使用 hsl 庫之後,採集 plc 時記憶體溢出,讓我幫忙看一下怎麼回事,哈哈,貌似是分析之旅中的第二次和 hsl 打交道,既然找到我,那就上 windbg 說話吧。 二:WinDbg 分析 1. 為什麼會記憶體溢出 簡單觀察程式的提交記憶體之 ...
  • 在 IIS 上啟用 Websocket 在 Windows Server 2012 或更高版本上啟用對 WebSocket 協議的支持: 備註 使用 IIS Express 時無需執行這些步驟 通過“管理”菜單或“伺服器管理器”中的鏈接使用“添加角色和功能”嚮導。 選擇“基於角色或基於功能的安裝”。 ...
  • C#-垃圾回收機制(GC) 什麼是GC 官網中有這麼一句話: The garbage collector is a common language runtime component that controls the allocation and release of managed memory ...
  • 呆了2個大屏行業的公司,對大屏幕有一些瞭解,所以整理下所瞭解的觸摸屏相關概念。方便自己以及進入這個行業的小伙伴們,能有個系統、快速的認知。 觸摸屏詳細的知識點,網上其實都有。整理資料過程中,我也瞭解了更多的觸摸屏知識,像聲波屏、光學屏之類的之前就沒接觸。下麵分不同的模塊,給大家介紹 交互觸摸屏類型 ...
  • 近段時間忙於各種項目和對【易排平臺】的優化,沒顧得上分享APS相關的小技巧,回頭看看小公眾號的關註人數早已達1500+,在此爭取時間寫一下這段時間在項目上及平臺優化過程中遇到的一些小技巧,以感謝諸位的關註。過去數月的解決的問題中,涉及最多的是規劃模型中,實現各種時間維度的功能,目前在平臺上也稍有成果 ...
  • 針對大量log日誌快速定位錯誤地方 動態查看日誌 tail -f catalina.ou 從頭打開日誌文件 cat catalina.ou 可以使用 >nanjiangtest.txt 輸出某個新日誌去查看 [[email protected] logs]# cat -n catalina.out |grep 7 ...
  • 前言 RocketMQ是阿裡巴巴旗下一款開源的MQ框架,經歷過雙十一考驗、Java編程語言實現,有非常好完整生態系統。RocketMQ作為一款純java、分散式、隊列模型的開源消息中間件,支持事務消息、順序消息、批量消息、定時消息、消息回溯等 本篇文章第一部分屬於一些核心概念和工作流程的講解;第二部 ...
  • 在java,c#類的成員修飾符包括,公有、私有、程式集可用的、受保護的。 對於python來說,只有兩個成員修飾符:公有成員,私有成員 成員修飾符是來修飾誰呢?當然是修飾成員了。那麼python類的成員包括什麼呢? python成員: 欄位,方法,屬性 每個類成員的修飾符有兩種: 公有成員:內部外部 ...