多图片多线程异步下载


开发中非常常用的就是就是图片下载,我们常用的就是SDWebImage,但是作为开发人员,不仅要能会用,还要知道其原理。本文就会介绍多图下载的实现。

本文中的示例Demno地址,下载后项目位于iOS_Demo/09-多图片多线程下载

1 - 问题

开篇我就先将多图下载过程中出现的问题摆出来,让我们能够一一克服

1:下载过程要放到子线程中,否则会导致UI线程卡顿

2:图片会重复下载,我们不仅要把下载完的图片放到内存中,还要把下载队列也缓存起来,否则可能图片没有下载完,会多次创建下载队列

3:沙盒缓存,放到哪里,根据下面对沙盒文件的介绍我们需要把图片放到Library/Caches目录下。

Document : iTunes会备份
Library :
1:Caches:缓存文件,不会清除
2:Preferences:偏好设置,保存账号信息
tmp :临时路径,随时会被删除,临时的不是重要的数据

下载流程图

明白了问题后,我们需要制定战略(画流程图),这样根据流程图,下载过程就更为清晰明了。

根据流程图一些坑,我们就可以完美的避免掉了。这样写程序也更有章法。

2 - 开始下载图片

根据前面的介绍,我们需要有两个缓存对象:

  1. 图片data的内存缓存
  2. 下载队列的缓存

我们可以使用NSMutableDictionary进行缓存,

这里我们假设两个缓存对象分别是:

/** 内存中图片的缓存字典 */
@property (nonatomic,strong)NSMutableDictionary * memoryImages; /** 下载操作的字典 */
@property (nonatomic,strong)NSMutableDictionary * operations;

2.1 内存缓存中取值

所以下载前,我们先要到内存中取值

   NSData *imageData = [self.memoryImages objectForKey:url];

如果imageData对象存在,我们就直接设置到Cell上。

2.2 内存缓存中没有,我们到沙盒中取值

内存缓存中不存在时我们到沙盒中进行查看:

        NSString *path = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES).lastObject;
//获取图片名字,图片名称不能包含URL,
NSString *fileName = [url lastPathComponent];
NSString *filePaPath = [path stringByAppendingPathComponent:fileName]; //内存中没有图片的缓存 -- 从沙盒中取出
NSData *imageData = [NSData dataWithContentsOfFile:filePaPath];

如果图片存在,我们需要把图片设置到Cell上,并且放一份到内存缓存中

//沙盒中取出后,放一份到内存缓存中
[self.memoryImages setObject:imageData forKey:url];

这样下次就不用频繁访问沙盒了。

2.2 内存缓存中没有,沙盒中也没有,我们需要判断下载队列是否已经存在

如果网速慢一些,并且Cell上下滑动的频率快一些的话,假如第一次第5行的Cell显示的时候,内存缓存中没有,沙盒中也没有,这时我们创建了下载对象,然后这行Cell被滑出屏幕,再次进来的时候图片还是没有下载完毕,经过判断内存缓存中没有,沙盒中也没有,我们如果再次创建下载对象的话,就会重复下载图片了,所以我们要把下载队列也进行缓存,当内存缓存中没有,沙盒中也没有时,我们先判断下载队列中是否已经存在了。

这里使用的是NSBlockOperation

NSBlockOperation *doo = [self.operations objectForKey:url];

2.2 内存缓存中没有,沙盒中也没有,下载队列也没有

这时我们就需要创建下载对象进行下载了。

首先我们先可以设置一个占位图,这样的显示更为友好一下。

 NSBlockOperation *downO = [NSBlockOperation blockOperationWithBlock:^{
NSURL *uurl = [NSURL URLWithString:url]; NSURLSession *session = [NSURLSession sharedSession];
NSURLSessionDataTask *down = [session dataTaskWithURL:uurl completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
//图片的下载过程 }]; [down resume];
}];

同时加入下载队列

  //下载对象加入队列中
[self.operations setObject:downO forKey:url];
[self.queue addOperation:downO];

图片下载完毕后,我们需要

//把图片data存入内存中
[self.memoryImages setObject:data forKey:url]; //同时写入沙盒中,永久缓存
[data writeToFile:filePaPath atomically:YES];

同时在主线程中更新UI

 [[NSOperationQueue mainQueue] addOperationWithBlock:^{

 UIImage *image = [UIImage imageWithData:data];
cell.iconView.image = image;
[self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
}];

这样完整的图片下载过程就好了。

【iOS系列】-多图片多线程异步下载的更多相关文章

  1. iOS开发网络篇—多线程断点下载

    iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件.因为实现过程较为复杂,所以下面贴出完整的代码. 实现思路:下载开始, ...

  2. IOS GCD图片数据异步下载,下载完成后合成显示

    关于GCD使用详解,请看我的上一篇blog:http://www.cnblogs.com/xin-lang/p/6278606.html 前段时间遇到个需要异步下载,下载完成后再组合显示的东西.这里我 ...

  3. 使用webClient实现图片同步,异步下载

    WebClient.DownloadFile 方法 将具有指定 URI 的资源下载到本地文件. 命名空间:System.Net 程序集:System(在 system.dll 中) 同步实现参考代码: ...

  4. iOS tableView的图片缓存异步载入

    1.建立一个viewController. .h文件实现UIScrollViewDelegate和UITableViewDelegate,并声明ICTableViewDelegate(用来实现图片有缓 ...

  5. iOS开发网络篇—大文件的多线程断点下载

    http://www.cnblogs.com/wendingding/p/3947550.html iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时 ...

  6. iOS开发网络篇—大文件的多线程断点下载(转)

    http://www.cnblogs.com/wendingding/p/3947550.html   iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了 ...

  7. [原]unity3d之http多线程异步资源下载

    郑重声明:转载请注明出处 U_探索 本文诞生于乐元素面试过程,被面试官问到AssetBundle多线程异步下载时,愣了半天,同样也被深深的鄙视一回(做了3年多u3d 这个都没用过),所以发誓要实现出来 ...

  8. 使用开源库 SDWebImage 异步下载缓存图片(持续更新)

    source  https://github.com/rs/SDWebImage APIdoc  http://hackemist.com/SDWebImage/doc Asynchronous im ...

  9. [Xcode 实际操作]八、网络与多线程-(22)使用GCD多线程技术异步下载图片

    目录:[Swift]Xcode实际操作 本文将演示如何使用使用GCD多线程技术异步下载图片. Grand Central Dispatch(GCD) 是 Apple 开发的一个多核编程的较新的解决方法 ...

随机推荐

  1. 安装Apache遇到的一点问题

    很久以前就安装好了Apache(2.2),现在再用时突然出现了问题: 以http://127.0.0.1/exercise/x.php的方式访问文件是正常的,但是要进入phpMyAdmin建表发现不能 ...

  2. Caffe学习系列(二)Caffe代码结构梳理,及相关知识点归纳

    前言: 通过检索论文.书籍.博客,继续学习Caffe,千里之行始于足下,继续努力.将自己学到的一些东西记录下来,方便日后的整理. 正文: 1.代码结构梳理 在终端下运行如下命令,可以查看caffe代码 ...

  3. 面向对象的全套“企业微信”api接口的代码实现,网上太多“面向过程”微信api接口的代码,这个开源给需要的人用

    有段时间没有写文章了. 一直以来,微信的热门是看得到的,很多人都需要与微信的api对接. 今天我这里就分享全套的企业微信api接口的代码. 关于微信api,网上已经有很多实现的了. 但是我今天之所以还 ...

  4. YIi2 Pjax简单使用

    1.点击事件需要在Pjax::begin() 和Pjax::end()范围内 2.需要在链接配置数组后加上  ['data-pjax'=>'#testPjax'] 其中,'#testPjax‘是 ...

  5. H5 Canvas vs. SVG

    HTML 5 Canvas vs. SVG HTML5 SVG HTML5 地理定位 Canvas 和 SVG 都允许您在浏览器中创建图形,但是它们在根本上是不同的. SVG SVG 是一种使用 XM ...

  6. 使用shape来定义控件的一些显示属性

    Android中常常使用shape来定义控件的一些显示属性,今天看了一些shape的使用,对shape有了大体的了解,稍作总结 先看下面的代码: <shape> <!-- 实心 -- ...

  7. python自学基础1week

    一.python老师介绍 二.为什么要学习python? 三.学习python有前途吗? 疗程1:语言基础 疗程2:网络编程 疗程3:web基础开发 疗程4:算法&设计模式 疗程5:pytho ...

  8. java学习笔记 --- 多态

    一.多态 (1)定义:同一个对象在不同时刻体现出来的不同状态.父类的引用或者接口的引用指向了自己的子类对象.   Dog d = new Dog();//Dog对象的类型是Dog类型.  Animal ...

  9. datagrid-detailview.js easyui表格嵌套

    datagrid-detailview.js easyui表格嵌套

  10. Failed to connect to Xilinx hw_server. Check if the hw_server is running and correct TCP port is used.

    Failed to connect to Xilinx hw_server. Check if the  hw_server is running and correct TCP port is us ...