iOS之SDWebImage原理

SDWebImage做爲目前最受歡迎的圖片下載第三方框架,使用率很高。可是你真的會用嗎?本文接下來將經過例子分析如何合理使用SDWebImage。html

使用場景:自定義的UITableViewCell上有圖片須要顯示,要求網絡網絡狀態爲WiFi時,顯示圖片高清圖;網絡狀態爲蜂窩移動網絡時,顯示圖片縮略圖。以下圖樣例:ios

1353118-c335652e795beb0c.jpg

圖中顯示的圖片符合根據網絡狀態下載要求緩存

  • 因爲要監聽網絡狀態,在這裏筆者推薦使用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被推出屏幕,此時緩存池爲空。

1461839550604205.png

cell尚未被推入緩存池

當有一個cell被推到屏幕以外時,系統會自動將這個cell放入自動緩存池。注意:cell對應的UIImage圖片數據模型並無清空!仍是指向上一個使用的cell。

1461839575796193.png

cell被放入緩存池

當下一個cell進入屏幕,系統會根據tableView註冊的標識找到對應的cell,拿來應用。上述進入緩存池的cell被從新添加進tableView,在tableView的Data Source方法tableView: cellForRowAtIndexPath:中設置改cell對應的模型數據,此時cell對應的如圖:

1461839596408204.png

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];
             }
         }
     }
相關文章
相關標籤/搜索