上篇文章给大家介绍了IOS多线程实现多图片下载1,本文继续给大家介绍ios多线程下载图片。
这次是用多线程进行图片的下载与存储,而且考虑到下载失败,占位图片的问题(第一张就是下载失败的图片)
闲话少说,上代码吧,因为有一部分和上次的一样,所以这里只上传不一样的
先给大家展示下效果图:
依旧都是在ViewController.m中
1.
1
2
3
4
5
6
7
8
9
10
|
@interface ViewController () //所有数据 @property (nonatomic,strong)NSArray *apps; //内存缓存图片 @property (nonatomic,strong)NSMutableDictionary *imgCache; /**所有操作*/ @property (nonatomic,strong)NSMutableDictionary *operations; /**队列对象*/ @property (nonatomic,strong) NSOperationQueue *queue; @end |
前两个和前面的一致
operations使用来存储下载图片的线程操作的字典,主要作用是防止重复下载
queue则是使用多线程时用到的队列
2.
1
2
3
4
5
6
7
8
|
- (NSOperationQueue *)queue { if (!_queue) { _queue = [[NSOperationQueue alloc] init]; //最大并发数 _queue.maxConcurrentOperationCount = 3; } return _queue; } |
对queue的初始化,以及控制子线程最多为3条
3.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *ID = @ "app" ; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; DDZApp *app = self.apps[indexPath.row]; cell.textLabel.text = app.name; cell.detailTextLabel.text = app.download; //先从内存中取出图片 UIImage *image = self.imgCache[app.icon]; if (image) { cell.imageView.image = image; } else { //内存中没有图片 //将图片文件数据写入到沙盒中 NSString *cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject]; //获得文件名 NSString *filename = [app.icon lastPathComponent]; //计算出文件的全路径 NSString *file = [cachesPath stringByAppendingPathComponent:filename]; //加载沙盒的文件数据 NSData *data = [NSData dataWithContentsOfFile:file]; //判断沙盒中是否有图片 if (data) { //直接加载沙盒中图片 UIImage *image = [UIImage imageWithData:data]; cell.imageView.image = image; //存到字典(内存)中 self.imgCache[app.icon] = image; } else { //下载图片 //占位图片 cell.imageView.image = [UIImage imageNamed:@ "place.jpg" ]; //先判断是否有下载任务 //加载失败后可以重复下载 NSOperation *operation = self.operations[app.icon]; if (operation == nil) { //这张图片没有下载任务 operation = [NSBlockOperation blockOperationWithBlock:^{ NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:app.icon]]; //数据加载失败 if (data == nil) { //移除操作 [self.operations removeObjectForKey:app.icon]; return ; } UIImage *image = [UIImage imageWithData:data]; //存到内存中 self.imgCache[app.icon] = image; //回到主线程显示图片 [[NSOperationQueue mainQueue] addOperationWithBlock:^{ //会出现重复占位的问题 //cell.imageView.image = image; //只需找到图片所在的行即可 [tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone]; }]; //将图片数据写入到沙盒中 [data writeToFile:file atomically:YES]; //移除操作 [self.operations removeObjectForKey:app.icon]; }]; //添加到下载队列 [self.queue addOperation:operation]; //添加到字典 self.operations[app.icon] = operation; } } } return cell; } |
这次绑定数据的方法内容有点多,因为考虑到了不少细节,不过逻辑和上次的差不多。