美文网首页
iOS图片多线程下载和缓存

iOS图片多线程下载和缓存

作者: 师从小马哥 | 来源:发表于2018-02-12 13:22 被阅读0次

前言

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    LearnTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier forIndexPath:indexPath];
    
//    [cell.img sd_setImageWithURL:[NSURL URLWithString:_dataList[indexPath.row]]];
    NSURL *url = [NSURL URLWithString:_dataList[indexPath.row]];
    NSData *imageData = [NSData dataWithContentsOfURL:url];
    UIImage *image = [UIImage imageWithData:imageData];
    
    cell.img.image = image;
    
    return cell;
}

初级加载网络图片方式, 具有以下缺点和解决方法:

  1. UI不流畅 -> 子线程下载图片
  2. 图片重复下载 -> 利用图片本地缓存
    2.1 使用字典进行内存缓存
    2.2 使用沙盒进行磁盘缓存

使用字典进行内存缓存

/** 内存缓存 */
@property (nonatomic, strong) NSMutableDictionary<NSString *, UIImage*> *images;

UIImage *image = [self.images objectForKey:appM.icon];
if (image) {
    cell.imageView.image = image;
    NSLog(@"%zd处的图片使用了内存缓存中的图片",indexPath.row) ;
} else {
    //把图片保存到内存缓存
    [self.images setObject:image forKey:<#UrlStr#>];
}

使用沙盒进行磁盘缓存

//保存图片到沙盒缓存
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
//获得图片的名称,不能包含/
NSString *fileName = [<#UrlStr#> lastPathComponent];
//拼接图片的全路径
NSString *fullPath = [caches stringByAppendingPathComponent:fileName];
//检查磁盘缓存
NSData *imageData = [NSData dataWithContentsOfFile:fullPath];
if (imageData) {
UIImage *image = [UIImage imageWithData:imageData];
cell.imageView.image = image;

NSLog(@"%zd处的图片使用了磁盘缓存中的图片",indexPath.row) ;
//把图片保存到内存缓存
[self.images setObject:image forKey:appM.icon];

} else {
        ....省略网络下载....
    //写数据到沙盒
    [imageData writeToFile:fullPath atomically:YES];
}

子线程下载图片

/** 队列 */
@property (nonatomic, strong) NSOperationQueue *queue;
-(NSOperationQueue *)queue
{
    if (_queue == nil) {
        _queue = [[NSOperationQueue alloc]init];
        //设置最大并发数
        _queue.maxConcurrentOperationCount = 5;
    }
    return _queue;
}
NSBlockOperation *download = [NSBlockOperation blockOperationWithBlock:^{
    NSURL *url = [NSURL URLWithString:_dataList[indexPath.row]];
    NSData *imageData = [NSData dataWithContentsOfURL:url];
    
    //写数据到沙盒
    [imageData writeToFile:fullPath atomically:YES];
    
    UIImage *image = [UIImage imageWithData:imageData];
    
    [NSThread sleepForTimeInterval:2.0];
    
    // 线程间通讯, 主线程刷新UI
    [[NSOperationQueue mainQueue] addOperationWithBlock:^{
        cell.img.image = image;
        NSLog(@"%zd处网络下载图片",indexPath.row);
    }];
}];
            
[self.queue addOperation:download];

多线程讨论:

  1. 这里的用到睡眠2秒钟来模拟网速差的情况, 如果不使用多线程, 程序会阻塞在主线程, 出现明显卡顿的现象.
  2. 上面代码还有一个隐蔽的缺陷, 例如当第一个cell 开始下载图片, 在两秒钟的时间内反复拖动cell 程序仍然会判断需要走下载图片流程. 继续优化....

线程操作做内存缓存

/** 操作缓存 */
@property (nonatomic, strong) NSMutableDictionary<NSString*, NSBlockOperation *> *operations;
-(NSMutableDictionary *)operations
{
    if (_operations == nil) {
        _operations = [NSMutableDictionary dictionary];
    }
    return _operations;
}

三步操作:

  1. 判断是否存在线程操作
  2. 添加线程操作
  3. 完成后移除操作
//检查该图片时候正在下载,如果是那么久什么都捕捉,否则再添加下载任务
NSBlockOperation *download = [self.operations objectForKey:_dataList[indexPath.row]];

if (download) {
    
} else {
    download = [NSBlockOperation blockOperationWithBlock:^{
        
        .....
        //移除图片的下载操作
        [self.operations removeObjectForKey:_dataList[indexPath.row]];
        ......
    }];
    
    //添加操作到操作缓存中
    [self.operations setObject:download forKey:_dataList[indexPath.row]];
......
}

还有两个bug问题:

  1. 由于cell重用, 图片错乱问题 -> 如果发现正在下载就清除图片或者加载占位图片
//检查该图片时候正在下载,如果是那么久什么都捕捉,否则再添加下载任务
NSBlockOperation *download = [self.operations objectForKey:lover.icon];
cell.img.image = [UIImage imageNamed:@"知道错了"];
if (download) {
    
}.....
  1. 执行闪退的问题
    图片添加内存缓存 图片为空值, 这个错误出现在网络加载图片后
[self.images setObject:image forKey:lover.icon];

解决方法就是判断为空 就及时 移除下载图片线程操作并return

download = [NSBlockOperation blockOperationWithBlock:^{
                    NSURL *url = [NSURL URLWithString:lover.icon];
                    NSData *imageData = [NSData dataWithContentsOfURL:url];
                    
                    //写数据到沙盒
                    [imageData writeToFile:fullPath atomically:YES];
                    
                    UIImage *image = [UIImage imageWithData:imageData];
                    
                    if (image == nil) {
                        //移除图片的下载操作
                        [self.operations removeObjectForKey:lover.icon];
                        return ;
                    }......

讨论:

以上是多图下载缓存的一个深入分析, 难点在于线程通讯, 还有一些细节处理

参考:
小马哥的多线程教程

相关文章

  • iOS多图片下载

    iOS多图片下载、在cell里面下载图片、做了缓存优化。 (app.icon是图片地址) // 先从内存缓存中取出...

  • iOS图片多线程下载和缓存

    前言 初级加载网络图片方式, 具有以下缺点和解决方法: UI不流畅 -> 子线程下载图片 图片重复下载 -> 利用...

  • iOS 开发中内存缓存与磁盘缓存?图片如何缓存?

    『导言』 iOS开发中,如何保证图片只被下载一次?如何缓存图片?内存缓存?磁盘缓存?到底如何区别?如何联系? 温馨...

  • iOS中SDWebImage框架浅析

    下载图片并显示 下载图片并计算下载进度 使用SDWebImageManager单例下载 同时做内存缓存和图片缓存可...

  • SDWebImage

    SDWebImage介绍 iOS中著名的网络图片处理框架. 包含的功能:图片下载,图片缓存,下载进度监听,GIF处...

  • 多线程03

    SDWebImage框架详解 下载图片并显示: 下载图片/显示图片/内存缓存/磁盘缓存 下载图片\内存缓存\磁盘缓...

  • iOS多线程之NSThread

    相关文章:iOS多线程之GCDiOS多线程之NSOperations 案例1--图片下载 案例2--多售票窗口同时...

  • SDWebImage框架的基本使用

    基本使用 1)下载图片并显示(内存缓存&磁盘缓存) 2)下载图片显示并计算下载进度(内存缓存&磁盘缓存&下载进度)...

  • SDWebImage

    一、简介 iOS中著名的牛逼的网络图片处理框架包含的功能:图片下载、图片缓存、下载进度监听、gif处理等等用法极其...

  • Swift开发-Kingfisher框架

    IOS开发-使用Kingfisher加载图片 Kingfisher (中文名:翠鸟) 是一个异步下载和缓存图片的库...

网友评论

      本文标题:iOS图片多线程下载和缓存

      本文链接:https://www.haomeiwen.com/subject/wvfltftx.html