iOS 緩存機制 (以SDWebImage 緩存圖片爲例)

在iOS開發過程中,經常要用到緩存機制,那麼如何較爲好的設計一個緩存機制呢?在這裏,小編以SDWebImage爲例,說一下這個緩存機制:

  • UIImageView+WebCache: setImageWithURL:placeholderImage:options: 先顯示 placeholderImage ,同時由SDWebImageManager 根據 URL 來在本地查找圖片。
  • SDWebImageManager: downloadWithURL:delegate:options:userInfo: SDWebImageManager是將UIImageView+WebCache同SDImageCache鏈接起來的類, SDImageCache: queryDiskCacheForKey:delegate:userInfo:用來從緩存根據CacheKey查找圖片是否已經在緩存中
  • 如果內存中已經有圖片緩存, SDWebImageManager會回調SDImageCacheDelegate : imageCache:didFindImage:forKey:userInfo:
  • 而UIImageView+WebCache 則回調SDWebImageManagerDelegate: webImageManager:didFinishWithImage:來顯示圖片。
  • 如果內存中沒有圖片緩存,那麼生成 NSInvocationOperation 添加到隊列,從硬盤查找圖片是否已被下載緩存。
  • 根據 URLKey 在硬盤緩存目錄下嘗試讀取圖片文件。這一步是在 NSOperation 進行的操作,所以回主線程進行結果回調 notifyDelegate:。
  • 如果上一操作從硬盤讀取到了圖片,將圖片添加到內存緩存中(如果空閒內存過小,會先清空內存緩存)。SDImageCacheDelegate 回調 imageCache:didFindImage:forKey:userInfo:。進而回調展示圖片。
  • 如果從硬盤緩存目錄讀取不到圖片,說明所有緩存都不存在該圖片,需要下載圖片,回調 imageCache:didNotFindImageForKey:userInfo:。
  • 共享或重新生成一個下載器 SDWebImageDownloader 開始下載圖片。
  • 圖片下載由 NSURLConnection 來做,實現相關 delegate 來判斷圖片下載中、下載完成和下載失敗。
  • connection:didReceiveData: 中利用 ImageIO 做了按圖片下載進度加載效果。
  • connectionDidFinishLoading: 數據下載完成後交給 SDWebImageDecoder 做圖片解碼處理。
  • 圖片解碼處理在一個 NSOperationQueue 完成,不會拖慢主線程 UI。如果有需要對下載的圖片進行二次處理,最好也在這裏完成,效率會好很多。
  • 在主線程 notifyDelegateOnMainThreadWithInfo: 宣告解碼完成,
  • imageDecoder:didFinishDecodingImage:userInfo: 回調給 SDWebImageDownloader。
  • imageDownloader:didFinishWithImage: 回調給 SDWebImageManager 告知圖片下載完成。
  • 通知所有的 downloadDelegates 下載完成,回調給需要的地方展示圖片。
  • 將圖片保存到 SDImageCache 中,內存緩存和硬盤緩存同時保存。
  • 寫文件到硬盤在單獨 NSInvocationOperation 中完成,避免拖慢主線程。
  • 如果是在iOS上運行,SDImageCache 在初始化的時候會註冊notification 到 UIApplicationDidReceiveMemoryWarningNotification 以及 UIApplicationWillTerminateNotification,在內存警告的時候清理內存圖片緩存,應用結束的時候清理過期圖片。
  • SDWebImagePrefetcher 可以預先下載圖片,方便後續使用。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章