SDWebImage下載圖片的底層實(shí)現(xiàn)

@WilliamAlex大叔

前言

SDWebImage框架是我們最常用的框架,我們下載圖片,清除緩存等都可以使用該框架.下面我們模仿SDWebImage來實(shí)現(xiàn)多圖片的下載

在寫代碼之前,我們先整理整理思路,下載圖片我們分為兩種情況,有緩存和沒緩存,下面我們來看看我做的兩幅圖

  • 無沙盒緩存


    無沙盒緩存.png
  • 有沙盒緩存

有沙盒緩存.png
  • 糾正一點(diǎn): 圖片中的下載操作是是保存到一個(gè)可變的字典中的,不是數(shù)組,畫圖的時(shí)候一整天都沒有吃飯,肚子太餓了,出了錯(cuò),還望大家見諒

代碼實(shí)現(xiàn)

在正式寫代碼之前,需要說明幾步操作

  • 定義模型類
  • 將storyboard中的控制器移除,拖入一個(gè)新的UITableViewController控制器,綁定ID:"apps"以及綁定控制器,將類型設(shè)置為subtitle,最后給控制器設(shè)置啟動(dòng)箭頭.
  • 將ViewController的父類換成UITableViewController
  • 代碼存在很大問題(需要優(yōu)化)
    • 問題 1,滑動(dòng)界面時(shí)有嚴(yán)重的卡頓現(xiàn)象,原因是下載操作是在主線程上執(zhí)行的.
#import "ViewController.h"
#import "WGApps.h"

@interface ViewController ()

/** 圖片緩存(plist文件中是字典存儲的) */
@property (nonatomic, weak) NSMutableDictionary *images;

/** apps數(shù)據(jù)源*/
@property(nonatomic, strong) NSArray *apps;

@end

@implementation ViewController

#pragma mark - 生命周期方法
- (void)viewDidLoad
{
    [super viewDidLoad];

    self.tableView.rowHeight = 44;
}


#pragma mark ----------------
#pragma mark - lazyLoading

- (NSMutableDictionary *)images {
    if (_images == nil) {

        _images = [NSMutableDictionary dictionary];
    }

    return _images;
}

- (NSArray *)apps {
    if (_apps == nil) {

        // 加載plist文件
        NSString *path = [[NSBundle mainBundle] pathForResource:@"apps.plist" ofType:nil];

        // 加載數(shù)據(jù)
        NSArray *dictArray = [NSArray arrayWithContentsOfFile:path];

        NSMutableArray *tempArray = [NSMutableArray arrayWithCapacity:dictArray.count];
        // 字典數(shù)組轉(zhuǎn)模型數(shù)組
        for (NSDictionary *dict in dictArray) {

         [tempArray addObject:[WGApps appsWithDict:dict]];
        }
        _apps = tempArray;
    }
    return _apps;
}

#pragma mark - 數(shù)據(jù)源方法

// 一共有多少個(gè)cell
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return self.apps.count;
}

// 每一個(gè)cell顯示什么內(nèi)容
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{

    // 定義ID(最好和storyboard中定義的ID一致)
    static NSString *ID = @"apps";

    // 創(chuàng)建cell
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];

    // 通過模型拿到對應(yīng)的資源
    WGApps *app = self.apps[indexPath.row];

    // 設(shè)置數(shù)據(jù)
    cell.textLabel.text = app.name;
    cell.detailTextLabel.text = app.download;

    // 下載頭像

    // 1, 首先去內(nèi)存緩存中取,如果沒有再去沙盒中去
    UIImage *image = [self.images objectForKey:app.icon];

    if (image) {
        // 來到這里,說明圖片緩存中已經(jīng)有需要的圖片,直接顯示到對應(yīng)的cell即可
        cell.imageView.image = image;
    } else
    {
        // 來到這里表示:圖片緩存中沒有所需要的圖片,那么這時(shí)候就要到對應(yīng)的沙盒中找有沒有下載的圖片

        // 1, 獲取沙盒路徑
        NSString *cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        // 注意:拿到沙盒路徑還不夠,我們需要的是下載圖片的路徑,所以需要拼接,拿到全路徑
        // 2, 獲取模型中圖片資源最后一個(gè)目錄名
        NSString *fileName = [app.icon lastPathComponent];

        // 3, 拼接路徑
        NSString *fullPath = [cachesPath stringByAppendingPathComponent:fileName];

        // 注意 : 在沙盒中的保存的資源是以二進(jìn)制的形式存在的.我們還需要判斷沙盒中是否有下載過的圖片
        // 4, 通過圖片路徑,拿到下載的圖片
        NSData *imageData = [NSData dataWithContentsOfFile:fullPath];

        // 5, 判斷沙盒中是否有值
        if (imageData) {
            // 來到這里說明沙盒中有下載好的圖片,直接將二進(jìn)制轉(zhuǎn)為圖片顯示即可,但是最后還需要講圖片報(bào)訊到圖片緩存中,方便下次直接獲取.
            UIImage *image = [UIImage imageWithData:imageData];

            // 顯示圖片
            cell.imageView.image = image;

            // 將圖片保存到圖片緩存中
            [self.images setObject:image forKey:app.icon];
        } else
        {

            // 來到這里說明沙盒中沒有值,這時(shí)候我們就需要下載圖片資源啦.
            // 下載圖片
            NSURL *url = [NSURL URLWithString:app.icon];

            // 將url轉(zhuǎn)為data保存到本地
            NSData *data = [NSData dataWithContentsOfURL:url];

            // 再將二進(jìn)制轉(zhuǎn)為圖片顯示到cell上
            UIImage *image = [UIImage imageWithData:data];

            // 顯示圖片
            cell.imageView.image = image;

            // 將下載的圖片保存到圖片緩存和沙盒中
            [self.images setObject:image forKey:app.icon];
            [data writeToFile:fullPath atomically:YES];

        }
    }
    return cell;
}

多圖片下載(優(yōu)化后的代碼)

  • 解決的問題 : 解決了卡頓現(xiàn)象,避免了重復(fù)下載

#import "ViewController.h"
#import "WGApps.h"


@interface ViewController ()

/** 圖片緩存(plist文件中是字典存儲的) */
@property (nonatomic, weak) NSMutableDictionary *images;

/** apps數(shù)據(jù)源*/
@property(nonatomic, strong) NSArray *apps;

/** 下載操作 */
@property(nonatomic, strong) NSMutableDictionary *operations;

/** 隊(duì)列 */
@property(nonatomic, strong) NSOperationQueue *queue;

@end

@implementation ViewController

#pragma mark - 生命周期方法
- (void)viewDidLoad
{
    [super viewDidLoad];

    self.tableView.rowHeight = 44;
}


#pragma mark ----------------
#pragma mark - lazyLoading

- (NSMutableDictionary *)operations
{
    if (_operations == nil) {
        _operations = [NSMutableDictionary dictionary];
    }
    return _operations;
}

- (NSOperationQueue *)queue {
    if (_queue == nil) {
        // 創(chuàng)建隊(duì)列
        _queue = [[NSOperationQueue alloc] init];

        // 設(shè)置最大并發(fā)數(shù)
        _queue.maxConcurrentOperationCount = 3;
    }
    return _queue;
}

- (NSMutableDictionary *)images {
    if (_images == nil) {

        _images = [NSMutableDictionary dictionary];
    }

    return _images;
}

- (NSArray *)apps {
    if (_apps == nil) {

        // 加載plist文件
        NSString *path = [[NSBundle mainBundle] pathForResource:@"apps.plist" ofType:nil];

        // 加載數(shù)據(jù)
        NSArray *dictArray = [NSArray arrayWithContentsOfFile:path];

        NSMutableArray *tempArray = [NSMutableArray arrayWithCapacity:dictArray.count];
        // 字典數(shù)組轉(zhuǎn)模型數(shù)組
        for (NSDictionary *dict in dictArray) {

         [tempArray addObject:[WGApps appsWithDict:dict]];
        }
        _apps = tempArray;
    }
    return _apps;
}

#pragma mark - 數(shù)據(jù)源方法

// 一共有多少個(gè)cell
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return self.apps.count;
}

// 每一個(gè)cell顯示什么內(nèi)容
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{

    // 定義ID(最好和storyboard中定義的ID一致)
    static NSString *ID = @"apps";

    // 創(chuàng)建cell
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];

    // 通過模型拿到對應(yīng)的資源
    WGApps *app = self.apps[indexPath.row];

    // 設(shè)置數(shù)據(jù)
    cell.textLabel.text = app.name;
    cell.detailTextLabel.text = app.download;

    // 下載頭像

    // 1, 首先去內(nèi)存緩存中取,如果沒有再去沙盒中去
    UIImage *image = [self.images objectForKey:app.icon];

    if (image) {
        // 來到這里,說明圖片緩存中已經(jīng)有需要的圖片,直接顯示到對應(yīng)的cell即可
        cell.imageView.image = image;
    } else
    {
        // 來到這里表示:圖片緩存中沒有所需要的圖片,那么這時(shí)候就要到對應(yīng)的沙盒中找有沒有下載的圖片

        // 1, 獲取沙盒路徑
        NSString *cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        // 注意:拿到沙盒路徑還不夠,我們需要的是下載圖片的路徑,所以需要拼接,拿到全路徑
        // 2, 獲取模型中圖片資源最后一個(gè)目錄名
        NSString *fileName = [app.icon lastPathComponent];

        // 3, 拼接路徑
        NSString *fullPath = [cachesPath stringByAppendingPathComponent:fileName];

        // 注意 : 在沙盒中的保存的資源是以二進(jìn)制的形式存在的.我們還需要判斷沙盒中是否有下載過的圖片
        // 4, 通過圖片路徑,拿到下載的圖片
        NSData *imageData = [NSData dataWithContentsOfFile:fullPath];

        // 5, 判斷沙盒中是否有值
        if (imageData) {
            // 來到這里說明沙盒中有下載好的圖片,直接將二進(jìn)制轉(zhuǎn)為圖片顯示即可,但是最后還需要講圖片報(bào)訊到圖片緩存中,方便下次直接獲取.
            UIImage *image = [UIImage imageWithData:imageData];

            // 顯示圖片
            cell.imageView.image = image;

            // 將圖片保存到圖片緩存中
            [self.images setObject:image forKey:app.icon];
        } else
        {

            //設(shè)置展占位圖片
            cell.imageView.image = [UIImage imageNamed:@"占位圖片"];

            //查看該圖片的下載操作是否存在
            NSBlockOperation *download = [self.operations objectForKey:app.icon];
            if (download == nil) {

                download = [NSBlockOperation blockOperationWithBlock:^{

                    // 下載圖片
                    NSURL *url = [NSURL URLWithString:app.icon];

                    // 阻塞1秒
                    [NSThread sleepForTimeInterval:1.0];

                    // 將圖片轉(zhuǎn)為二進(jìn)制保存到本地沙盒中
                    NSData *data = [NSData dataWithContentsOfURL:url];

                    // 將二進(jìn)制轉(zhuǎn)為圖片顯示
                    UIImage *image = [UIImage imageWithData:data];

                    // 如果沒有圖片,一定要講下載操作從字典中移除
                    if (image == nil) {
                        [self.operations removeObjectForKey:app.icon];
                        return ;
                    }

                    //保存圖片到內(nèi)存緩存
                    [self.images setObject:image forKey:app.icon];

                    //保存圖片到沙河緩存
                    [data writeToFile:fullPath atomically:YES];

                    //線程間通信
                    [[NSOperationQueue mainQueue]addOperationWithBlock:^{

                        //刷新cell
                        [tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
                    }];
                    [self.operations removeObjectForKey:app.icon];
                }];

                //加入到操作緩存
                [self.operations setObject:download forKey:app.icon];

                //把操作添加到隊(duì)列
                [self.queue addOperation:download];
            }
        }
    }
    return cell;
}

@end

我們使用框架來實(shí)現(xiàn)同樣功能

  • 以上就是模仿SDWebImage內(nèi)部實(shí)現(xiàn)多圖片下載的原理,接下來我們使用SDWebImage來實(shí)現(xiàn)多圖片下載
#import "ViewController.h"
#import "WGApps.h"
#import "UIImageView+WebCache.h"


@interface ViewController ()

/** apps數(shù)據(jù)源*/
@property(nonatomic, strong) NSArray *apps;

@end

@implementation ViewController

#pragma mark - 生命周期方法
- (void)viewDidLoad
{
    [super viewDidLoad];

    self.tableView.rowHeight = 84;
}

#pragma mark ----------------
#pragma mark - lazyLoading

- (NSArray *)apps {
    if (_apps == nil) {

        // 加載plist文件
        NSString *path = [[NSBundle mainBundle] pathForResource:@"apps.plist" ofType:nil];

        // 加載數(shù)據(jù)
        NSArray *dictArray = [NSArray arrayWithContentsOfFile:path];

        NSMutableArray *tempArray = [NSMutableArray arrayWithCapacity:dictArray.count];
        // 字典數(shù)組轉(zhuǎn)模型數(shù)組
        for (NSDictionary *dict in dictArray) {

         [tempArray addObject:[WGApps appsWithDict:dict]];
        }
        _apps = tempArray;
    }
    return _apps;
}

#pragma mark - 數(shù)據(jù)源方法

// 一共有多少個(gè)cell
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return self.apps.count;
}

// 每一個(gè)cell顯示什么內(nèi)容
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{

    // 定義ID(最好和storyboard中定義的ID一致)
    static NSString *ID = @"apps";

    // 創(chuàng)建cell
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];

    // 通過模型拿到對應(yīng)的資源
    WGApps *app = self.apps[indexPath.row];

    // 設(shè)置數(shù)據(jù)
    cell.textLabel.text = app.name;
    cell.detailTextLabel.text = app.download;

    // 下載圖片
    [cell.imageView sd_setImageWithURL:[NSURL URLWithString:app.icon] placeholderImage:[UIImage imageNamed:@"placehoder"]];

    return cell;
}

@end

總結(jié)

  • 從上面的代碼來看,使用第三方框架大大減少了我們的工作量,不過在使用第三方框架時(shí),我們還是最好多了解一點(diǎn)框架的內(nèi)部原理,這樣即使出錯(cuò)了,我們很快就找到問題所在,就像上面優(yōu)化代碼時(shí),我遇到了一個(gè)bug : 我在拼接絕對路徑時(shí),使用錯(cuò)了方法,本應(yīng)該使用:stringByAppendingPathComponent.但是我的粗心使用了 : stringByAppendingString,導(dǎo)致出現(xiàn)了問題.經(jīng)過這樣去了解,這種bug我以后是不會再犯了,即使犯了,也會很快改正過來o
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末亿笤,一起剝皮案震驚了整個(gè)濱河市齐媒,隨后出現(xiàn)的幾起案子匆笤,更是在濱河造成了極大的恐慌,老刑警劉巖扫夜,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件楞泼,死亡現(xiàn)場離奇詭異,居然都是意外死亡笤闯,警方通過查閱死者的電腦和手機(jī)堕阔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來颗味,“玉大人超陆,你說我怎么就攤上這事∑致恚” “怎么了时呀?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長晶默。 經(jīng)常有香客問我谨娜,道長,這世上最難降的妖魔是什么磺陡? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任趴梢,我火速辦了婚禮漠畜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘坞靶。我一直安慰自己憔狞,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布滩愁。 她就那樣靜靜地躺著躯喇,像睡著了一般。 火紅的嫁衣襯著肌膚如雪硝枉。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天倦微,我揣著相機(jī)與錄音妻味,去河邊找鬼。 笑死欣福,一個(gè)胖子當(dāng)著我的面吹牛责球,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播拓劝,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼雏逾,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了郑临?” 一聲冷哼從身側(cè)響起栖博,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎厢洞,沒想到半個(gè)月后仇让,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡躺翻,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年丧叽,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片公你。...
    茶點(diǎn)故事閱讀 39,902評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡踊淳,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出陕靠,到底是詐尸還是另有隱情迂尝,我是刑警寧澤,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布懦傍,位于F島的核電站雹舀,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏粗俱。R本人自食惡果不足惜说榆,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一虚吟、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧签财,春花似錦串慰、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至神汹,卻和暖如春庆捺,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背屁魏。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工滔以, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人氓拼。 一個(gè)月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓你画,卻偏偏與公主長得像,于是被迫代替她去往敵國和親桃漾。 傳聞我的和親對象是個(gè)殘疾皇子坏匪,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評論 2 354

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

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫、插件适滓、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,103評論 4 62
  • 一般建筑工程位置都較為偏遠(yuǎn),為了方便建筑工程相關(guān)人員的生活几于,公司都會建造一個(gè)項(xiàng)目部生活區(qū)蕊苗。生活區(qū)里是螞蟻雖小五臟...
    茂繁閱讀 526評論 10 21
  • 我們?nèi)タ礋熁鸷脝幔咳デ迫幔タ茨窃旃被ㄖ腥绾卧偕被ǖ构龋瑝艟持先绾卧佻F(xiàn)夢境渤愁。讓我們并肩走過荒涼的河岸仰望夜空抖格,生命的...
    虛線閱讀 207評論 0 0