Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android圖片緩存原理、特性對比

Android圖片緩存原理、特性對比

編輯:關於Android編程

這是我在 MDCC 上分享的內容(略微改動),也是源碼解析第一期發布時介紹的源碼解析後續會慢慢做的事。

從總體設計和原理上對幾個圖片緩存進行對比,沒用到他們的朋友也可以了解他們在某些特性上的實現。

一. 四大圖片緩存基本信息


Universal ImageLoader 是很早開源的圖片緩存,在早期被很多應用使用。

Picasso 是 Square 開源的項目,且他的主導者是 JakeWharton,所以廣為人知。

Glide 是 Google 員工的開源項目,被一些 Google App 使用,在去年的 Google I/O 上被推薦,不過目前國內資料不多。

Fresco 是 Facebook 在今年上半年開源的圖片緩存,主要特點包括:
(1) 兩個內存緩存加上 Native 緩存構成了三級緩存

(2) 支持流式,可以類似網頁上模糊漸進式顯示圖片

(3) 對多幀動畫圖片支持更好,如 Gif、WebP

鑒於 Fresco 還沒發布正式的 1.0 版本,同時一直沒太多時間熟悉 Fresco 源碼,後面對比不包括 Fresco,以後有時間再加入對比。

更多圖片緩存庫可見:Android 圖片緩存庫

二、基本概念

在正式對比前,先了解幾個圖片緩存通用的概念:
(1) RequestManager:請求生成和管理模塊

(2) Engine:引擎部分,負責創建任務(獲取數據),並調度執行

(3) GetDataInterface:數據獲取接口,負責從各個數據源獲取數據。
比如 MemoryCache 從內存緩存獲取數據、DiskCache 從本地緩存獲取數據,下載器從網絡獲取數據等。

(4) Displayer:資源(圖片)顯示器,用於顯示或操作資源。
比如 ImageView,這幾個圖片緩存都不僅僅支持 ImageView,同時支持其他 View 以及虛擬的 Displayer 概念。

(5) Processor 資源(圖片)處理器
負責處理資源,比如旋轉、壓縮、截取等。

以上概念的稱呼在不同圖片緩存中可能不同,比如 Displayer 在 ImageLoader 中叫做 ImageAware,在 Picasso 和 Glide 中叫做 Target。

三、共同優點

1. 使用簡單
都可以通過一句代碼可實現圖片獲取和顯示。

2. 可配置度高,自適應程度高
圖片緩存的下載器(重試機制)、解碼器、顯示器、處理器、內存緩存、本地緩存、線程池、緩存算法等大都可輕松配置。

自適應程度高,根據系統性能初始化緩存配置、系統信息變更後動態調整策略。
比如根據 CPU 核數確定最大並發數,根據可用內存確定內存緩存大小,網絡狀態變化時調整最大並發數等。

3. 多級緩存
都至少有兩級緩存、提高圖片加載速度。 

4. 支持多種數據源
支持多種數據源,網絡、本地、資源、Assets 等

5. 支持多種 Displayer
不僅僅支持 ImageView,同時支持其他 View 以及虛擬的 Displayer 概念。

其他小的共同點包括支持動畫、支持 transform 處理、獲取 EXIF 信息等。

四、ImageLoader 設計及優點

1. 總體設計及流程

上面是 ImageLoader 的總體設計圖。整個庫分為 ImageLoaderEngine,Cache 及 ImageDownloader,ImageDecoder,BitmapDisplayer,BitmapProcessor 五大模塊,其中 Cache 分為 MemoryCache 和 DiskCache 兩部分。

簡單的講就是 ImageLoader 收到加載及顯示圖片的任務,並將它交給 ImageLoaderEngine,ImageLoaderEngine 分發任務到具體線程池去執行,任務通過 Cache 及 ImageDownloader 獲取圖片,中間可能經過 BitmapProcessor 和 ImageDecoder 處理,最終轉換為Bitmap 交給 BitmapDisplayer 在 ImageAware 中顯示。

2. ImageLoader 優點

(1) 支持下載進度監聽

(2) 可以在 View 滾動中暫停圖片加載
通過 PauseOnScrollListener 接口可以在 View 滾動中暫停圖片加載。

(3) 默認實現多種內存緩存算法 這幾個圖片緩存都可以配置緩存算法,不過 ImageLoader 默認實現了較多緩存算法,如 Size 最大先刪除、使用最少先刪除、最近最少使用、先進先刪除、時間最長先刪除等。

(4) 支持本地緩存文件名規則定義

五、Picasso 設計及優點

1. 總體設計及流程

上面是 Picasso 的總體設計圖。整個庫分為 Dispatcher,RequestHandler 及 Downloader,PicassoDrawable 等模塊。

Dispatcher 負責分發和處理 Action,包括提交、暫停、繼續、取消、網絡狀態變化、重試等等。

簡單的講就是 Picasso 收到加載及顯示圖片的任務,創建 Request 並將它交給 Dispatcher,Dispatcher 分發任務到具體 RequestHandler,任務通過 MemoryCache 及 Handler(數據獲取接口) 獲取圖片,圖片獲取成功後通過 PicassoDrawable 顯示到 Target 中。

需要注意的是上面 Data 的 File system 部分,Picasso 沒有自定義本地緩存的接口,默認使用 http 的本地緩存,API 9 以上使用 okhttp,以下使用 Urlconnection,所以如果需要自定義本地緩存就需要重定義 Downloader。

2. Picasso 優點

(1) 自帶統計監控功能
支持圖片緩存使用的監控,包括緩存命中率、已使用內存大小、節省的流量等。

(2) 支持優先級處理
每次任務調度前會選擇優先級高的任務,比如 App 頁面中 Banner 的優先級高於 Icon 時就很適用。

(3) 支持延遲到圖片尺寸計算完成加載

(4) 支持飛行模式、並發線程數根據網絡類型而變
手機切換到飛行模式或網絡類型變換時會自動調整線程池最大並發數,比如 wifi 最大並發為 4, 4g 為 3,3g 為 2。
這裡 Picasso 根據網絡類型來決定最大並發數,而不是 CPU 核數。 

(5) “無”本地緩存
無”本地緩存,不是說沒有本地緩存,而是 Picasso 自己沒有實現,交給了 Square 的另外一個網絡庫 okhttp 去實現,這樣的好處是可以通過請求 Response Header 中的 Cache-Control 及 Expired 控制圖片的過期時間。

六、Glide 設計及優點

1. 總體設計及流程

上面是 Glide 的總體設計圖。整個庫分為 RequestManager(請求管理器),Engine(數據獲取引擎)、 Fetcher(數據獲取器)、MemoryCache(內存緩存)、DiskLRUCache、Transformation(圖片處理)、Encoder(本地緩存存儲)、Registry(圖片類型及解析器配置)、Target(目標) 等模塊。

簡單的講就是 Glide 收到加載及顯示資源的任務,創建 Request 並將它交給RequestManager,Request 啟動 Engine 去數據源獲取資源(通過 Fetcher ),獲取到後 Transformation 處理後交給 Target。

Glide 依賴於 DiskLRUCache、GifDecoder 等開源庫去完成本地緩存和 Gif 圖片解碼工作。

2. Glide 優點

(1) 圖片緩存->媒體緩存
Glide 不僅是一個圖片緩存,它支持 Gif、WebP、縮略圖。甚至是 Video,所以更該當做一個媒體緩存。 

(2) 支持優先級處理

(3) 與 Activity/Fragment 生命周期一致,支持 trimMemory
Glide 對每個 context 都保持一個 RequestManager,通過 FragmentTransaction 保持與 Activity/Fragment 生命周期一致,並且有對應的 trimMemory 接口實現可供調用。

(4) 支持 okhttp、Volley
Glide 默認通過 UrlConnection 獲取數據,可以配合 okhttp 或是 Volley 使用。實際 ImageLoader、Picasso 也都支持 okhttp、Volley。

(5) 內存友好
① Glide 的內存緩存有個 active 的設計
從內存緩存中取數據時,不像一般的實現用 get,而是用 remove,再將這個緩存數據放到一個 value 為軟引用的 activeResources map 中,並計數引用數,在圖片加載完成後進行判斷,如果引用計數為空則回收掉。

② 內存緩存更小圖片
Glide 以 url、view_width、view_height、屏幕的分辨率等做為聯合 key,將處理後的圖片緩存在內存緩存中,而不是原始圖片以節省大小

③ 與 Activity/Fragment 生命周期一致,支持 trimMemory

④ 圖片默認使用默認 RGB_565 而不是 ARGB_888
雖然清晰度差些,但圖片更小,也可配置到 ARGB_888。

其他:Glide 可以通過 signature 或不使用本地緩存支持 url 過期

七、匯總


三者總體上來說,ImageLoader 的功能以及代理容易理解長度都一般。 

Picasso 代碼雖然只在一個包下,沒有嚴格的包區分,但代碼簡單、邏輯清晰,一兩個小時就能叫深入的了解完。

Glide 功能強大,但代碼量大、流轉復雜。在較深掌握的情況下才推薦使用,免得出了問題難以下手解決。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved