ios开发多线程四:NSOperation多图下载综合案例
#import "ViewController.h"
#import "XMGAPP.h" @interface ViewController ()
/** tableView的数据源 */
@property (nonatomic, strong) NSArray *apps;
/** 内存缓存 */
@property (nonatomic, strong) NSMutableDictionary *images;
/** 队列 */
@property (nonatomic, strong) NSOperationQueue *queue;
/** 操作缓存 */
@property (nonatomic, strong) NSMutableDictionary *operations;
@end @implementation ViewController #pragma mark ----------------------
#pragma mark lazy loading
-(NSOperationQueue *)queue
{
if (_queue == nil) {
_queue = [[NSOperationQueue alloc]init];
//设置最大并发数
_queue.maxConcurrentOperationCount = ;
}
return _queue;
}
-(NSMutableDictionary *)images
{
if (_images == nil) {
_images = [NSMutableDictionary dictionary];
}
return _images;
}
-(NSArray *)apps
{
if (_apps == nil) { //字典数组
NSArray *arrayM = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle]pathForResource:@"apps.plist" ofType:nil]]; //字典数组---->模型数组
NSMutableArray *arrM = [NSMutableArray array];
for (NSDictionary *dict in arrayM) {
[arrM addObject:[XMGAPP appWithDict:dict]];
}
_apps = arrM;
}
return _apps;
} -(NSMutableDictionary *)operations
{
if (_operations == nil) {
_operations = [NSMutableDictionary dictionary];
}
return _operations;
} #pragma mark ----------------------
#pragma mark UITableViewDatasource
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
return ;
} -(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return self.apps.count;
} -(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *ID = @"app"; //1.创建cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; //2.设置cell的数据
//2.1 拿到该行cell对应的数据
XMGAPP *appM = self.apps[indexPath.row]; //2.2 设置标题
cell.textLabel.text = appM.name; //2.3 设置子标题
cell.detailTextLabel.text = appM.download; //2.4 设置图标 //先去查看内存缓存中该图片时候已经存在,如果存在那么久直接拿来用,否则去检查磁盘缓存
//如果有磁盘缓存,那么保存一份到内存,设置图片,否则就直接下载
//1)没有下载过
//2)重新打开程序 UIImage *image = [self.images objectForKey:appM.icon];
if (image) {
cell.imageView.image = image;
NSLog(@"%zd处的图片使用了内存缓存中的图片",indexPath.row) ;
}else
{
//保存图片到沙盒缓存
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
//获得图片的名称,不能包含/
NSString *fileName = [appM.icon lastPathComponent];
//拼接图片的全路径
NSString *fullPath = [caches stringByAppendingPathComponent:fileName]; //检查磁盘缓存
NSData *imageData = [NSData dataWithContentsOfFile:fullPath];
// //废除
// imageData = nil; if (imageData) {
UIImage *image = [UIImage imageWithData:imageData];
cell.imageView.image = image; NSLog(@"%zd处的图片使用了磁盘缓存中的图片",indexPath.row) ;
//把图片保存到内存缓存
[self.images setObject:image forKey:appM.icon]; // NSLog(@"%@",fullPath);
}else
{
//检查该图片时候正在下载,如果是那么久什么都捕捉,否则再添加下载任务
NSBlockOperation *download = [self.operations objectForKey:appM.icon];
if (download) { }else
{ //先清空cell原来的图片
cell.imageView.image = [UIImage imageNamed:@"Snip20160221_306"]; download = [NSBlockOperation blockOperationWithBlock:^{
NSURL *url = [NSURL URLWithString:appM.icon];
NSData *imageData = [NSData dataWithContentsOfURL:url];
UIImage *image = [UIImage imageWithData:imageData]; NSLog(@"%zd--下载---",indexPath.row); //容错处理
if (image == nil) {
[self.operations removeObjectForKey:appM.icon];
return ;
}
//演示网速慢的情况
//[NSThread sleepForTimeInterval:3.0]; //把图片保存到内存缓存
[self.images setObject:image forKey:appM.icon]; //NSLog(@"Download---%@",[NSThread currentThread]);
//线程间通信
[[NSOperationQueue mainQueue] addOperationWithBlock:^{ //cell.imageView.image = image;
//刷新一行
[self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationLeft];
//NSLog(@"UI---%@",[NSThread currentThread]);
}]; //写数据到沙盒
[imageData writeToFile:fullPath atomically:YES]; //移除图片的下载操作
[self.operations removeObjectForKey:appM.icon]; }]; //添加操作到操作缓存中
[self.operations setObject:download forKey:appM.icon]; //添加操作到队列中
[self.queue addOperation:download];
} }
} //3.返回cell
return cell;
} -(void)didReceiveMemoryWarning
{
[self.images removeAllObjects]; //取消队列中所有的操作
[self.queue cancelAllOperations];
} //1.UI很不流畅 --- > 开子线程下载图片
//2.图片重复下载 ---> 先把之前已经下载的图片保存起来(字典)
//内存缓存--->磁盘缓存 //3.图片不会刷新--->刷新某行
//4.图片重复下载(图片下载需要时间,当图片还未完全下载之前,又要重新显示该图片)
//5.数据错乱 ---设置占位图片 /*
Documents:会备份,不允许
Libray
Preferences:偏好设置 保存账号
caches:缓存文件
tmp:临时路径(随时会被删除)
*/ /**
* 1:在项目中读取项目中的文件:
NSArray *arrayM = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle]pathForResource:@"apps.plist" ofType:nil]];
读取url;arrayWithContentsOfUrl 2:kvc:快速实现字典转模型,前提必须得保证模型中的字段值一一对应
+(instancetype)appWithDict:(NSDictionary *)dict
{
XMGAPP *appM = [[XMGAPP alloc]init];
//KVC
[appM setValuesForKeysWithDictionary:dict];
return appM;
} 3:设计思路:1:将下载的图片分别存储在内存缓存和磁盘缓存中,然后先去内存缓存中查找有无图片,若内存缓存中没有图片(有图片则就直接设置图片),则去磁盘缓存中查找图片中,若无图片,则就去执行下载操作,也需要将每张图片的下载操作存到内存缓存中,下载完成后,将图片分别存到内存与磁盘缓存中,并从内存缓存中移除操作队列,有图片,直接设置,并存到内存缓存中。2:内存缓存:就是一个强引用的属性变量,一般用字典去进行缓存,键值对一一对应 磁盘缓存:就是缓存到沙盒, Documents:会在itools上备份,苹果不许可在Documents下存储缓存,上线会被拒
Libray:一般Libray用于存储缓存信息,大文件或是离线缓存的数据,它包括以下的两个路径
Preferences:偏好设置 保存账号
caches:缓存文件
tmp:临时路径(随时会被删除) 4:具体实现:1:NSCachesDirectory:获得Libray下的caches文件夹路径,[appM.icon lastPathComponent]获得文件的扩展名,一般将文件的扩展名和路径拼接起来作为文件名 //保存图片到沙盒缓存
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
//获得图片的名称,不能包含/
NSString *fileName = [appM.icon lastPathComponent];
//拼接图片的全路径
NSString *fullPath = [caches stringByAppendingPathComponent:fileName]; 2:检查磁盘缓存
NSData *imageData = [NSData dataWithContentsOfFile:fullPath]; 3:磁盘缓存有图片,就设置,并缓存到内存缓存中,无图片直接下载,先从缓存队列中找出缓存队列,如果有什么都不做,没有,就去创建队列,封装任务NSBlockOperation,(每张图片创建一个队列,并将队列缓存到内存缓存中),线程间通信,子主线程刷新UI,此时将下载的图片缓存到内存与磁盘缓存中,将队列缓存到内存缓存中,以便下次继续使用。
4:还需要有容错处理,当图片下载失败后,直接return返回,并移除缓存中的队列
if (image == nil) {
[self.operations removeObjectForKey:appM.icon];
return ;
} 5:遇到的问题以及解决办法: //1.UI很不流畅 --- > 开子线程下载图片
//2.图片重复下载 ---> 先把之前已经下载的图片保存起来(字典)内存缓存--->磁盘缓存
//3.图片不会刷新--->刷新某行:[self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationLeft];
//4.图片重复下载(图片下载需要时间,当图片还未完全下载之前,又要重新显示该图片,快速的溢出屏幕再移回来,通过设置缓存队列来解决问题,让之前正在下载的队列一直还在处理下载任务)
//5.数据错乱 ---设置占位图片:考虑到cell的复用机制,当有一个图片从屏幕顶端移出屏幕后,会放到缓存池,从底端进入后,从缓存池中取出图片,此时显示的图片还是缓存池中cell的那个图片,下载完成后才会更新,解决办法是将图片更新前设为nil,效果不好,最好是设置占位图片 */ @end
#import <Foundation/Foundation.h> @interface XMGAPP : NSObject /** APP的名称 */
@property (nonatomic, strong) NSString *name;
/** APP的图片的url地址 */
@property (nonatomic, strong) NSString *icon;
/** APP的下载量 */
@property (nonatomic, strong) NSString *download; +(instancetype)appWithDict:(NSDictionary *)dict;
@end
#import "XMGAPP.h" @implementation XMGAPP +(instancetype)appWithDict:(NSDictionary *)dict
{
XMGAPP *appM = [[XMGAPP alloc]init];
//KVC
[appM setValuesForKeysWithDictionary:dict]; return appM;
}
@end
补充:https在plist中的配置:
###3.多图下载综合示例程序
(1)涉及知识点
01 字典转模型
02 存储数据到沙盒,从沙盒中加载数据
03 占位图片的设置(cell的刷新问题)
04 如何进行内存缓存(使用NSDictionary)
05 在程序开发过程中的一些容错处理
06 如何刷新tableView的指定行(解决数据错乱问题)
07 NSOperation以及线程间通信相关知识
ios开发多线程四:NSOperation多图下载综合案例的更多相关文章
- iOS开发多线程篇—NSOperation基本操作
iOS开发多线程篇—NSOperation基本操作 一.并发数 (1)并发数:同时执⾏行的任务数.比如,同时开3个线程执行3个任务,并发数就是3 (2)最大并发数:同一时间最多只能执行的任务的个数. ...
- iOS开发多线程篇—NSOperation简单介绍
iOS开发多线程篇—NSOperation简单介绍 一.NSOperation简介 1.简单说明 NSOperation的作⽤:配合使用NSOperation和NSOperationQueue也能实现 ...
- iOS开发——多线程篇——NSOperation(基于GCD多线程编程),下载图片并合成新图片
一.NSOperation的基本概念1.简介NSOperation的作用配合使用NSOperation和NSOperationQueue也能实现多线程编程 NSOperation和NSOperatio ...
- iOS开发多线程--(NSOperation/Queue)
iOS实现多线程的方式有三种,分别是NSThread.NSOperation.GCD. 关于GCD,请阅读GCD深入浅出学习 简介 NSOperation封装了需要执行的操作和执行操作所需的数据,提供 ...
- IOS开发 多线程编程 - NSOperation
一.NSOperation 1.简介 NSOperation实例封装了需要执行的操作和执行操作所需的数据,并且能够以并发或非并发的方式执行这个操作. NSOperation本身是抽象基类,因此必须使用 ...
- iOS开发多线程篇 10 —NSOperation基本操作
iOS开发多线程篇—NSOperation基本操作 一.并发数 (1)并发数:同时执⾏行的任务数.比如,同时开3个线程执行3个任务,并发数就是3 (2)最大并发数:同一时间最多只能执行的任务的个数. ...
- iOS开发多线程篇 09 —NSOperation简单介绍
iOS开发多线程篇—NSOperation简单介绍 一.NSOperation简介 1.简单说明 NSOperation的作⽤:配合使用NSOperation和NSOperationQueue也能实现 ...
- iOS开发多线程篇—自定义NSOperation
iOS开发多线程篇—自定义NSOperation 一.实现一个简单的tableView显示效果 实现效果展示: 代码示例(使用以前在主控制器中进行业务处理的方式) 1.新建一个项目,让控制器继承自UI ...
- iOS开发多线程篇 11 —自定义NSOperation
iOS开发多线程篇—自定义NSOperation 一.实现一个简单的tableView显示效果 实现效果展示: 代码示例(使用以前在主控制器中进行业务处理的方式) 1.新建一个项目,让控制器继承自UI ...
随机推荐
- MySQL具体解释(13)------------事务
一. 什么是事务 事务就是一段sql 语句的批处理.可是这个批处理是一个atom(原子) .不可切割,要么都运行,要么回滚(rollback)都不运行. 二.为什么出现这样的技术 为什么要使用事务这个 ...
- js24---工厂模式2
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/stri ...
- CMake设置生成vs工程的动态库输出路径
作者:朱金灿 来源:http://blog.csdn.net/clever101 在网上搜了很多的资料,发现CMake不能设置一个动态库工程的输出目录和中间目录,难道除了VC之外其它编译器如gcc中没 ...
- JavaScript学习总结(2)——JavaScript数据类型判断
最近做项目中遇到了一些关于javascript数据类型的判断处理,上网找了一下资料,并且亲自验证了各种数据类型的判断,在此做一个总结吧! 一.JS中的数据类型 1.数值型(Number):包括整数. ...
- [Javascript AST] 4. Continue: Report ESLint error
const disallowedMethods = ["log", "info", "warn", "error", & ...
- Skill of vim
用vim也有一段时间了,谨以此记下一些有意思的技巧. 跳转 hjkl,左下右上. b/e跳转到上/下一个单词 ^/$跳转到行头/尾 从{,[,(,相应的},],)相互跳转.能够用% 跳转到局部变量的定 ...
- secureCRT The remote system refused the connection问题解决
问题: Ubuntu系统必须开启ssh服务后,XP或者其它的主机才干够远程登陆到Ubuntu系统. 1,安装软件包,运行sudo apt-get install openssh-server Ubun ...
- meld文件的脚本
今天模仿着别人的脚本,结合网上的资料,摸索着写了一个简单的脚本,用来打开meld 工具.这个脚本虽然简单,但这是第一次自己写脚本,记录下来,作为自己python学习的起点.代码如下 #/use/bin ...
- 21.Spring Boot 使用Java代码创建Bean并注册到Spring中
转自:https://blog.csdn.net/catoop/article/details/50558333
- 从大整数乘法的实现到 Karatsuba 快速算法
Karatsuba 快速乘积算法是具有独特合并过程(combine/merge)的分治算法(Karatsuba 是俄罗斯人).此算法主要是对两个整数进行相乘,并不适用于低位数(如 int 的 32 位 ...