SDWebImage做爲目前最受歡迎的圖片下載第三方框架,使用率很高。可是你真的會用嗎?本文接下來將經過例子分析如何合理使用SDWebImage。html
使用場景:自定義的UITableViewCell上有圖片須要顯示,要求網絡網絡狀態爲WiFi時,顯示圖片高清圖;網絡狀態爲蜂窩移動網絡時,顯示圖片縮略圖。以下圖樣例:ios
圖中顯示的圖片符合根據網絡狀態下載要求緩存
因爲要監聽網絡狀態,在這裏筆者推薦使用AFNetWorking。網絡
1)在GitHub或者利用cocoaPod給項目導入第三方框架AFNetWorking。app
2)在AppDelegate.m文件中的application:didFinishLaunchingWithOptions:方法中監聽網絡狀態。框架
// AppDelegate.m 文件中
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
// 監控網絡狀態
[[AFNetworkReachabilityManager sharedManager] startMonitoring];
}
// 如下代碼在須要監聽網絡狀態的方法中使用
AFNetworkReachabilityManager *mgr = [AFNetworkReachabilityManager sharedManager];
if
(mgr.isReachableViaWiFi) {
// 在使用Wifi, 下載原圖
}
else
{
// 其餘,下載小圖
}
}
這時就會有iOS學習者開始抱怨:這不是很簡單嗎?因而三下五除二寫完了如下代碼。async
// 利用MVC,在設置cell的模型屬性時候,下載圖片
- setItem:(CustomItem *)item
{
_item = item;
UIImage *placeholder = [UIImage imageNamed:@
"placeholderImage"
];
AFNetworkReachabilityManager *mgr = [AFNetworkReachabilityManager sharedManager];
if
(mgr.isReachableViaWiFi) {
// 在使用Wifi, 下載原圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
{
// 其餘,下載小圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.thumbnailImage] placeholderImage:placeholder];
}
}
此時,確實能完成基本的按照當前網絡狀態下載對應的圖片,可是真實開發中,這樣實際上是不合理的。如下是須要注意的細節:學習
1)SDWebImage會自動幫助開發者緩存圖片(包括內存緩存,沙盒緩存),因此咱們須要設置用戶在WiFi環境下下載的高清圖,下次在蜂窩網絡狀態下打開應用也應顯示高清圖,而不是去下載縮略圖。url
2)許多應用設置模塊帶有一個功能:移動網絡環境下仍然顯示高清圖。這個功能實際上是將設置記錄在沙盒中,關於數據保存到本地,能夠查看本人另外一篇簡書首頁文章:iOS本地數據存取,看這裏就夠了。spa
3)當用戶處於離線狀態時候,沒法合理處理業務。
因而,開始加以改進。爲了讓讀者你更容易理解,我先貼出僞代碼:
- setItem:(CustomItem *)item
{
_item = item;
if
(緩存中有原圖)
{
self.imageView.image = 原圖;
}
else
{
if
(Wifi環境)
{
下載顯示原圖
}
else
if
(手機自帶網絡)
{
if
(3G\4G環境下仍然下載原圖)
{
下載顯示原圖
}
else
{
下載顯示小圖
}
}
else
{
if
(緩存中有小圖)
{
self.imageView.image = 小圖;
}
else
// 處理離線狀態
{
self.imageView.image = 佔位圖片;
}
}
}
}
實現上面的僞代碼:讀者能夠一一對應上面的僞代碼。練習的時候推薦先寫僞代碼,再寫真實代碼
多多注意「註釋」解釋。
- setItem:(CustomItem *)item
{
_item = item;
// 佔位圖片
UIImage *placeholder = [UIImage imageNamed:@
"placeholderImage"
];
// 從內存\沙盒緩存中得到原圖,
UIImage *originalImage = [[SDImageCache sharedImageCache] imageFromDiskCacheForKey:item.originalImage];
if
(originalImage) {
// 若是內存\沙盒緩存有原圖,那麼就直接顯示原圖(無論如今是什麼網絡狀態)
self.imageView.image = originalImage;
}
else
{
// 內存\沙盒緩存沒有原圖
AFNetworkReachabilityManager *mgr = [AFNetworkReachabilityManager sharedManager];
if
(mgr.isReachableViaWiFi) {
// 在使用Wifi, 下載原圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
if
(mgr.isReachableViaWWAN) {
// 在使用手機自帶網絡
// 用戶的配置項假設利用NSUserDefaults存儲到了沙盒中
// [[NSUserDefaults standardUserDefaults] setBool:NO forKey:@"alwaysDownloadOriginalImage"];
// [[NSUserDefaults standardUserDefaults] synchronize];
#warning 從沙盒中讀取用戶的配置項:在3G\4G環境是否仍然下載原圖
BOOL alwaysDownloadOriginalImage = [[NSUserDefaults standardUserDefaults] boolForKey:@
"alwaysDownloadOriginalImage"
];
if
(alwaysDownloadOriginalImage) {
// 下載原圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
{
// 下載小圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.thumbnailImage] placeholderImage:placeholder];
}
}
else
{
// 沒有網絡
UIImage *thumbnailImage = [[SDImageCache sharedImageCache] imageFromDiskCacheForKey:item.thumbnailImage];
if
(thumbnailImage) {
// 內存\沙盒緩存中有小圖
self.imageView.image = thumbnailImage;
}
else
{
// 處理離線狀態,並且有沒有緩存時的狀況
self.imageView.image = placeholder;
}
}
}
}
解決了嗎?真正的坑纔剛剛開始。
在表述上述代碼的坑以前,咱們先來分析一下UITableViewCell的緩存機制。
請看下圖:有一個tableView正在同時顯示三個UITableViewCell,每一個tableViewCell包含一個imageView的子控件,並且每一個cell都有一個對應的模型屬性用來設置imageView的圖片內容。
注意:因爲沒有cell被推出屏幕,此時緩存池爲空。
cell尚未被推入緩存池
當有一個cell被推到屏幕以外時,系統會自動將這個cell放入自動緩存池。注意:cell對應的UIImage圖片數據模型並無清空!仍是指向上一個使用的cell。
cell被放入緩存池
當下一個cell進入屏幕,系統會根據tableView註冊的標識找到對應的cell,拿來應用。上述進入緩存池的cell被從新添加進tableView,在tableView的Data Source方法tableView: cellForRowAtIndexPath:中設置改cell對應的模型數據,此時cell對應的如圖:
cell被放入緩存池
以上就是tableView重用機制的簡單介紹。
從新回來,那麼上面所說的真正的坑在哪呢?
用一個場景來描述一下吧:當用戶所處環境WiFi網速不夠快(不能當即將圖片下載完畢),而在上述代碼,在WiFi環境下又是下載高清大圖。因此須要必定的時間來完成下載。而就在此時,用戶不肯等,想看看上次打開App時顯示的圖片,此時用戶會滑處處於下面的cell來查看。注意:此時,上面的cell下載圖片操做並無暫停,還在處於下載圖片狀態中。當用戶在查看上次打開App的顯示圖片時(上次打開App下載完成的圖片,SDWebImage會幫咱們緩存,不用下載),而正好須要顯示上次打開App時的圖片的cell是利用tableView重用機制而從緩存池中拿出來的cell,等處處於上面的cell的高清大圖已經下載好了的時候,SDWebImage默認作法是,立馬把下載好的圖片設置給ImageView,因此咱們這時候會在底下的顯示的cell顯示上面的圖片,形成數據錯亂,這是很是嚴重的BUG。
那麼該如何解決這個棘手的問題呢?
若是咱們能在cell被從緩存池中拿出來使用的時候,將這個cell放入緩存池以前的下載操做移除,那麼就不會出現數據錯亂了。
這時候你可能會問我:怎麼移除下載操做?下載操做不是SDWebImage幫咱們作的嗎?
說的沒錯,確實是SDWebImage幫咱們下載圖片的,咱們來扒一扒SDWebImage源碼,看看他是怎麼完成的。
- (void)sd_setImageWithURL:(NSURL *)url placeholderImage:(UIImage *)placeholder options:(SDWebImageOptions)options progress:(SDWebImageDownloaderProgressBlock)progressBlock completed:(SDWebImageCompletionBlock)completedBlock {
// 關閉當前圖片的下載操做
[self sd_cancelCurrentImageLoad];
objc_setAssociatedObject(self, &imageURLKey, url, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
if
(!(options & SDWebImageDelayPlaceholder)) {
dispatch_main_async_safe(^{
self.image = placeholder;
});
}
if
(url) {
// check if activityView is enabled or not
if
([self showActivityIndicatorView]) {
[self addActivityIndicator];
}
__weak __typeof(self)wself = self;
id operation = [SDWebImageManager.sharedManager downloadImageWithURL:url options:options progress:progressBlock completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, BOOL finished, NSURL *imageURL) {
[wself removeActivityIndicator];
if
(!wself)
return
;
dispatch_main_sync_safe(^{
if
(!wself)
return
;
if
(image && (options & SDWebImageAvoidAutoSetImage) && completedBlock)
{
completedBlock(image, error, cacheType, url);
return
;
}
else
if
(image) {
wself.image = image;
[wself setNeedsLayout];
}
else
{
if
((options & SDWebImageDelayPlaceholder)) {
wself.image = placeholder;
[wself setNeedsLayout];
}
}
if
(completedBlock && finished) {
completedBlock(image, error, cacheType, url);
}
});
}];
[self sd_setImageLoadOperation:operation forKey:@
"UIImageViewImageLoad"
];
}
else
{
dispatch_main_async_safe(^{
[self removeActivityIndicator];
if
(completedBlock) {
NSError *error = [NSError errorWithDomain:SDWebImageErrorDomain code:-1 userInfo:@{NSLocalizedDescriptionKey : @
"Trying to load a nil url"
}];
completedBlock(nil, error, SDImageCacheTypeNone, url);
}
});
}
}
咱們驚奇的發現,原來SDWebImage在下載圖片時,第一件事就是關閉imageView當前的下載操做!
是否是開始感嘆SDWebImage多麼神奇了?沒錯,咱們只須要把咱們寫的那段代碼全部的直接訪問本地緩存代碼利用SDWebImage進行設置就OK了!
下面就是完成版代碼。
- setItem:(CustomItem *)item
{
_item = item;
// 佔位圖片
UIImage *placeholder = [UIImage imageNamed:@
"placeholderImage"
];
// 從內存\沙盒緩存中得到原圖
UIImage *originalImage = [[SDImageCache sharedImageCache] imageFromDiskCacheForKey:item.originalImage];
if
(originalImage) {
// 若是內存\沙盒緩存有原圖,那麼就直接顯示原圖(無論如今是什麼網絡狀態)
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
{
// 內存\沙盒緩存沒有原圖
AFNetworkReachabilityManager *mgr = [AFNetworkReachabilityManager sharedManager];
if
(mgr.isReachableViaWiFi) {
// 在使用Wifi, 下載原圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
if
(mgr.isReachableViaWWAN) {
// 在使用手機自帶網絡
// 用戶的配置項假設利用NSUserDefaults存儲到了沙盒中
// [[NSUserDefaults standardUserDefaults] setBool:NO forKey:@"alwaysDownloadOriginalImage"];
// [[NSUserDefaults standardUserDefaults] synchronize];
#warning 從沙盒中讀取用戶的配置項:在3G\4G環境是否仍然下載原圖
BOOL alwaysDownloadOriginalImage = [[NSUserDefaults standardUserDefaults] boolForKey:@
"alwaysDownloadOriginalImage"
];
if
(alwaysDownloadOriginalImage) {
// 下載原圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.originalImage] placeholderImage:placeholder];
}
else
{
// 下載小圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.thumbnailImage] placeholderImage:placeholder];
}
}
else
{
// 沒有網絡
UIImage *thumbnailImage = [[SDImageCache sharedImageCache] imageFromDiskCacheForKey:item.thumbnailImage];
if
(thumbnailImage) {
// 內存\沙盒緩存中有小圖
[self.imageView sd_setImageWithURL:[NSURL URLWithString:item.thumbnailImage] placeholderImage:placeholder];
}
else
{
[self.imageView sd_setImageWithURL:nil placeholderImage:placeholder];
}
}
}