iOS 10如何優(yōu)雅地異步加載圖片

newYear.png

前言:在 APP 開發(fā)中時常需要展示大量的圖片,然而加載顯示照片是不僅消耗APP性能镇眷,同時在tableView的cell中上下滑動加載圖片也是一項非常耗時和耗流量的動作。

本文主要分兩個部分:

1.嘗試自己實現(xiàn)異步加載圖片

2.如何優(yōu)雅異步加載圖片

嘗試自己實現(xiàn)異步加載圖片

首先創(chuàng)建一個UIImageView的category,先將 UIImageView 的 image 設(shè)置為placeholder戒努,然后發(fā)出網(wǎng)絡(luò)請求剔交,獲取圖片肆饶;如果圖片獲取成功就在主線程中將UIImageView的image替換成請求成功的照片。

#import <UIKit/UIKit.h>

@interface UIImageView (YY)

- (void)yy_setImageWithURL:(NSURL *)url placeholderImage:(UIImage *)placeholder ;

@end
#import "UIImageView+YY.h"
@implementation UIImageView (YY)

- (void)yy_setImageWithURL:(NSURL *)url placeholderImage:(UIImage *)placeholder
{
    self.image = placeholder;
  dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSData *data = [NSData dataWithContentsOfURL:url];
        NSLog(@"data:%@",data);
        if (data) {
            dispatch_async(dispatch_get_main_queue(), ^{
                UIImage *image = [UIImage imageWithData:data];
                [self setImage:image];
            });
        }
    });
}
@end

通過上面自己的實現(xiàn)方式岖常,對于UIImageView可以加載并顯示圖片的驯镊。但是如果UIImageView設(shè)置了多次加載圖片,會出現(xiàn)怎樣的結(jié)果呢竭鞍?

#import "ViewController.h"
#import "UIImageView+YY.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    
    NSLog(@"沙盒:%@",NSHomeDirectory());
    
    UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(200, 200, 100, 100)];
    [self.view addSubview:imageView];
    [imageView yy_setImageWithURL:[NSURL URLWithString:@"https://image.png"] placeholderImage:[UIImage imageNamed:@"景區(qū)列表Grid_03"]];
    [imageView yy_setImageWithURL:[NSURL URLWithString:@"https://image.png"] placeholderImage:[UIImage imageNamed:@"景區(qū)列表Grid_03"]];
    [imageView yy_setImageWithURL:[NSURL URLWithString:@"https://image.png"] placeholderImage:[UIImage imageNamed:@"景區(qū)列表Grid_03"]];
}
@end

當(dāng)多次為同一個UIImageView設(shè)置image的時候板惑,根據(jù)上面的情況,我們最終需要顯示的會是image3偎快,但是這時候UIImageView會去開啟三個下載任務(wù)冯乘,但由于他們關(guān)聯(lián)的都會是同一個UIImageView,就會出現(xiàn)3個任務(wù)的回調(diào)都給同一個UIImageView晒夹,這樣就會出現(xiàn)相同數(shù)據(jù)來刷新UI了裆馒。既不能有良好的用戶體驗,也浪費用戶的流量丐怯。接下來我們就嘗試解決這個問題喷好。

嘗試自己實現(xiàn)異步加載圖片

大概實現(xiàn)思路:這次異步加載圖片我們使用的是NSOperation,而不是GCD读跷。

然后需要兩個字典類型的數(shù)據(jù):一個( images)是用于記錄已經(jīng)下載了的圖片 避免再次下載, URL 為 key梗搅,另一個(operations)是用于記錄用于下載的圖片, URL 為 key。

在加載圖片之前效览,我們首先查找images中是否在內(nèi)存中已經(jīng)存在即將需要下載的圖片无切,如果有,則直接通過鍵值對的方式將圖片取出丐枉;當(dāng)圖片不存在 于內(nèi)存中哆键,我們再去沙盒中查找是否已經(jīng)保存,當(dāng)前面兩中方式都無法查找需要的圖片的時候矛洞,這才正式開始通過網(wǎng)絡(luò)請求異步加載圖片洼哎。值得注意的是,在我們通過網(wǎng)絡(luò)異步加載圖片的時候沼本,我們需要通過operations來判斷是不是已經(jīng)存在噩峦。

定義NSOperationQueue,用于存放下載操作:

@property (nonatomic,strong) NSOperationQueue    *operationQueue;

定義所需要的兩個字典類型的變量:

//用于記錄已經(jīng)下載了的圖片 避免再次下載 URL 為 key
@property (nonatomic,strong) NSMutableDictionary *images;
//用于記錄用于下載的圖片 URL 為 key
@property (nonatomic,strong) NSMutableDictionary *operations;

接下來需要去判斷內(nèi)存和沙盒中是否已經(jīng)存在需要下載的圖片:

UIImage *imageTmp = self.images[url];//UIImage *imageCache = [self.images valueForKey:(NSString *)url]; && [self.images objectForKey:(NSString *)url];
    if (imageTmp) {
        completeBlock(imageTmp);
        return;
    }
    //沙盒
    NSString *imagePath = imageFile(url);
    NSData *imageCache = [NSData dataWithContentsOfFile:imagePath];
    if (imageCache) {
        UIImage *image = [UIImage imageWithData:imageCache];
        completeBlock(image);
        return;
    }

如果內(nèi)存和沙盒中都沒有需要的照片抽兆,那么即將進入網(wǎng)路請求识补,在進行網(wǎng)絡(luò)請求的時候,先判斷NSOperationQueue中是否已經(jīng)存在下載操作:

[self.operations enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        if ([key isEqualToString:(NSString *)url]) {
            //該URL 正在下載  --->已經(jīng)取消正在下載的 然后重新下載(還是保持正在下載的 暫停即將要下載的 )
            return;
        }
    }];

如果上面的所有情況都沒有辫红,那么久需要進行網(wǎng)絡(luò)請求加載圖片:

//該 URL 沒有在下載隊列中 就將該 URL 加入下載隊列
    __weak typeof(self) weakself = self;
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        __strong typeof(weakself) strongSelf = weakself;
        NSData *data = [NSData dataWithContentsOfURL:url];
        if (data) {
            UIImage *image = [UIImage imageWithData:data];
            if (strongSelf) {
                self.images[url] = image;
            }
            //將下載好的圖片存放于沙盒中
            NSData* ImageData = UIImagePNGRepresentation(image);
            [ImageData writeToFile:imageFile(url) atomically:YES];
            
            [[NSOperationQueue mainQueue]addOperationWithBlock:^{
                if (strongSelf) {
                    [self.operations removeObjectForKey:url];
                }
                completeBlock(image);//更新 UI 要回到主線程
            }];
        }
    }];
    
    self.operations[url] = operation;
    [self.operationQueue addOperation:operation];

以上大概講述了怎么去異步下載圖片凭涂,但是在真正的項目中并不能發(fā)揮什么作用。因為強大的開源項目SDWebImage已經(jīng)很優(yōu)雅有效地解決了這些問題贴妻。以上是實現(xiàn)思路也是參考了SDWebImage切油,當(dāng)初步看了以上的實現(xiàn)思路,再去看SDWebImage的開源代碼名惩,思路也會清晰容易一些澎胡。SDWebImage在代碼細節(jié)和思路上的實現(xiàn),非常值得我們?nèi)W(xué)習(xí)娩鹉。

總結(jié):只要代碼思維邏輯實現(xiàn)得好攻谁,緩存實現(xiàn)得妥當(dāng),肯定不會卡弯予,面對大量圖片顯示的APP戚宦,既能優(yōu)化用戶的體驗,也能省流量锈嫩。即使手機不能上網(wǎng)受楼,打開app時依然顯示數(shù)據(jù)和圖片,保證用戶體驗呼寸。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末那槽,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子等舔,更是在濱河造成了極大的恐慌骚灸,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件慌植,死亡現(xiàn)場離奇詭異甚牲,居然都是意外死亡,警方通過查閱死者的電腦和手機蝶柿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進店門丈钙,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人交汤,你說我怎么就攤上這事雏赦〗袤希” “怎么了?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵星岗,是天一觀的道長填大。 經(jīng)常有香客問我,道長俏橘,這世上最難降的妖魔是什么允华? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮寥掐,結(jié)果婚禮上靴寂,老公的妹妹穿的比我還像新娘。我一直安慰自己召耘,他們只是感情好百炬,可當(dāng)我...
    茶點故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著污它,像睡著了一般收壕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上轨蛤,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天蜜宪,我揣著相機與錄音,去河邊找鬼祥山。 笑死圃验,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的缝呕。 我是一名探鬼主播澳窑,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼供常!你這毒婦竟也來了摊聋?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤栈暇,失蹤者是張志新(化名)和其女友劉穎麻裁,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體源祈,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡煎源,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了香缺。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片手销。...
    茶點故事閱讀 39,902評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖图张,靈堂內(nèi)的尸體忽然破棺而出锋拖,到底是詐尸還是另有隱情诈悍,我是刑警寧澤,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布兽埃,位于F島的核電站侥钳,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏讲仰。R本人自食惡果不足惜慕趴,卻給世界環(huán)境...
    茶點故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一痪蝇、第九天 我趴在偏房一處隱蔽的房頂上張望鄙陡。 院中可真熱鬧,春花似錦躏啰、人聲如沸趁矾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽毫捣。三九已至,卻和暖如春帝际,著一層夾襖步出監(jiān)牢的瞬間蔓同,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工蹲诀, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留斑粱,地道東北人。 一個月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓脯爪,卻偏偏與公主長得像则北,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子痕慢,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,843評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫尚揣、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,103評論 4 62
  • 前不久做了一個生成快照的需求掖举,其中用到 SDWebImage 來下載圖片快骗,在使用該框架的過程中也遇到了一些問題,索...
    ShannonChenCHN閱讀 14,068評論 12 241
  • 先母生平坎坷塔次,歷盡磨難滨巴,性情謙和,訥言謹行俺叭,懷珍福音恭取,安貧樂善,教養(yǎng)兒孫熄守,幾忘其身蜈垮。母生于豫省耗跛,幼習(xí)針黹,其技揚名...
    西山鋤藥閱讀 1,513評論 0 1
  • 時間真的非常寶貴攒发,一眨眼一天就過去了调塌,我決定給自己的時間定個目標(biāo),定個向惠猿。 1羔砾、日精進還是放在晚上比較好,這樣方便...
    超級奶爸閱讀 652評論 1 2
  • 備料 炒制 等待 完成
    小糖米1004閱讀 247評論 0 0