添加存儲(chǔ)路徑
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
// Override point for customization after application launch.
//添加一個(gè)通用的只讀存儲(chǔ)路徑
NSString * bundlePath = [[[NSBundlemainBundle]resourcePath] stringByAppendingPathComponent:@"CustomPathImages"];
[[SDImageCachesharedImageCache]addReadOnlyCachePath:bundlePath];
return YES;
}
第一步,下載SDWebImage,導(dǎo)入工程。github托管地址https://github.com/rs/SDWebImage
第二步,在需要的地方導(dǎo)入頭文件
#import "UIImageView+WebCache.h"
SDWebImage是一個(gè)很厲害的圖片緩存的框架。既ASIHttp+AsyncImage之后,我一直使用AFNetworking集成的UIImageView+AFNetworking.h,但后者對(duì)于圖片的緩存實(shí)際應(yīng)用的是NSURLCache自帶的cache機(jī)制。而NSURLCache每次都要把緩存的raw data 再轉(zhuǎn)化為UIImage,就帶來(lái)了數(shù)據(jù)處理和內(nèi)存方面的更多操作。具體的比較在這里。
SDWebImage提供了如下三個(gè)category來(lái)進(jìn)行緩存。
MKAnnotationView(WebCache)
UIButton(WebCache)
UIImageView(WebCache)
以最為常用的UIImageView為例:
UIImageView+WebCache: setImageWithURL:placeholderImage:options:
先顯示 placeholderImage ,同時(shí)由SDWebImageManager 根據(jù) URL 來(lái)在本地查找圖片。
SDWebImageManager: downloadWithURL:delegate:options:userInfo:
SDWebImageManager是將UIImageView+WebCache同SDImageCache鏈接起來(lái)的類(lèi), SDImageCache: queryDiskCacheForKey:delegate:userInfo:
用來(lái)從緩存根據(jù)CacheKey查找圖片是否已經(jīng)在緩存中
如果內(nèi)存中已經(jīng)有圖片緩存, SDWebImageManager會(huì)回調(diào)SDImageCacheDelegate : imageCache:didFindImage:forKey:userInfo:
而 UIImageView+WebCache 則回調(diào)SDWebImageManagerDelegate: webImageManager:didFinishWithImage:
來(lái)顯示圖片。
如果內(nèi)存中沒(méi)有圖片緩存,那么生成 NSInvocationOperation 添加到隊(duì)列,從硬盤(pán)查找圖片是否已被下載緩存。
根據(jù) URLKey 在硬盤(pán)緩存目錄下嘗試讀取圖片文件。這一步是在 NSOperation 進(jìn)行的操作,所以回主線(xiàn)程進(jìn)行結(jié)果回調(diào) notifyDelegate:
。
如果上一操作從硬盤(pán)讀取到了圖片,將圖片添加到內(nèi)存緩存中(如果空閑內(nèi)存過(guò)小,會(huì)先清空內(nèi)存緩存)。SDImageCacheDelegate 回調(diào) imageCache:didFindImage:forKey:userInfo:
。進(jìn)而回調(diào)展示圖片。
如果從硬盤(pán)緩存目錄讀取不到圖片,說(shuō)明所有緩存都不存在該圖片,需要下載圖片,回調(diào) imageCache:didNotFindImageForKey:userInfo:
。
共享或重新生成一個(gè)下載器 SDWebImageDownloader
開(kāi)始下載圖片。
圖片下載由 NSURLConnection 來(lái)做,實(shí)現(xiàn)相關(guān) delegate 來(lái)判斷圖片下載中、下載完成和下載失敗。
connection:didReceiveData:
中利用 ImageIO 做了按圖片下載進(jìn)度加載效果。
connectionDidFinishLoading:
數(shù)據(jù)下載完成后交給 SDWebImageDecoder
做圖片解碼處理。
圖片解碼處理在一個(gè) NSOperationQueue 完成,不會(huì)拖慢主線(xiàn)程 UI。如果有需要對(duì)下載的圖片進(jìn)行二次處理,最好也在這里完成,效率會(huì)好很多。
在主線(xiàn)程 notifyDelegateOnMainThreadWithInfo:
宣告解碼完成,imageDecoder:didFinishDecodingImage:userInfo:
回調(diào)給 SDWebImageDownloader。
imageDownloader:didFinishWithImage:
回調(diào)給 SDWebImageManager 告知圖片下載完成。
通知所有的 downloadDelegates 下載完成,回調(diào)給需要的地方展示圖片。
將圖片保存到 SDImageCache 中,內(nèi)存緩存和硬盤(pán)緩存同時(shí)保存。
寫(xiě)文件到硬盤(pán)在單獨(dú) NSInvocationOperation 中完成,避免拖慢主線(xiàn)程。
如果是在iOS上運(yùn)行,SDImageCache 在初始化的時(shí)候會(huì)注冊(cè)notification 到 UIApplicationDidReceiveMemoryWarningNotification 以及 UIApplicationWillTerminateNotification,在內(nèi)存警告的時(shí)候清理內(nèi)存圖片緩存,應(yīng)用結(jié)束的時(shí)候清理過(guò)期圖片。
SDWebImagePrefetcher
可以預(yù)先下載圖片,方便后續(xù)使用。