iOS多线程自定义operation加载图片 不重复下载图片
摘要:1:ios通过抽象类NSOperation封装了gcd,让ios的多线程变得更为简单易用;
2:耗时的操作交给子线程来完成,主线程负责ui的处理,提示用户的体验
2:自定义operation继承自NSOperation,在子线程中下载图片;
3:保证图片只下载一次,还有保证下载任务不重复
------------------------------------------------------------------------------------
实现原理:1:图片缓存:用字典保存图片和图片的url,key:url value:图片
2:任务缓存:用字典保存operation任务和图片的url,key:url value:operation
先看自定义的operation:.h文件
1 #import <Foundation/Foundation.h> 2 @class XBOperation; 3 4 @protocol XBOperationDelegate <NSObject> 5 6 -(void)operation:(XBOperation *)operation didFinishDownloadImage:(UIImage *)image; 7 8 @end 9 /** 10 * 自定义operation 11 */ 12 @interface XBOperation : NSOperation 13 14 // 代理 15 @property (nonatomic, weak)id<XBOperationDelegate> delegate; 16 17 /** 18 * url 19 */ 20 @property (nonatomic, copy) NSString *url; 21 /** 22 * indexPath 和tableview中的cell相对应 23 */ 24 @property (nonatomic, strong) NSIndexPath *indexPath; 25 26 @end
说明:设置代理的目的是等完成下载图片的后通知控制器拿图片
自定义operation的.m文件
@implementation XBOperation -(void)main { if (self.isCancelled) return; NSURL *downloadUrl = [NSURL URLWithString:self.url]; NSData *data = [NSData dataWithContentsOfURL:downloadUrl]; // 这行会比较耗时 if (self.isCancelled) return; UIImage *image = [UIImage imageWithData:data]; if (self.isCancelled) return; NSLog(@"--%@--", [NSThread currentThread]); if ([self.delegate respondsToSelector:@selector(operation:didFinishDownloadImage:)]) { dispatch_async(dispatch_get_main_queue(), ^{ // 回到主线程, 传递图片数据给代理对象 [self.delegate operation:self didFinishDownloadImage:image]; }); } } @end
说明:重写main方法,在把任务operation加入到队列后,队列会自动调用任务的start方法,start方法内部会调用main方法来完成操作
控制器的tableviewcell数据源的实现
#pragma mark dataSource数据源方法 -(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView { return 1; } -(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return self.apps.count; } // 数据源方法 实现cell中的内容填充 -(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *ID = @"app"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; if (!cell) { cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:ID]; } // 设置数据 XBApp是数据模型 XBApp *app = self.apps[indexPath.row]; // 设置标题和子标题 cell.textLabel.text = app.name; cell.detailTextLabel.text = app.download; cell.imageView.image = [UIImage imageNamed:@"57437179_42489b0"]; // 图片数据设置 UIImage *image = self.imageCache[app.icon]; if (image) { // 如果图片缓存中有图片 直接加载到 cell.imageView.image = image; }else{ // 如果缓存中没有数据 // 先设置占位占位图片 cell.imageView.image = [UIImage imageNamed:@"57437179_42489b0"]; // 创建自定义的operation 先根据当前要显示图片的url从队列中找到operation 看是否正在下载中 XBOperation *operationDown = self.operationQueue[app.icon]; if (!operationDown) { // 如果下载operation不存在 就创建 并添加到队列中 operationDown = [[XBOperation alloc] init]; operationDown.delegate = self; // 设置代理 operationDown.indexPath = indexPath; // 把每个cell与一个operation绑定 operationDown.url = app.icon; // 把operation添加到队列中 会自动调用start方法 然后调用operation的main的方法 [self.queue addOperation:operationDown]; // 把operation这个下载任务添加到operation的字典中 防止重复下载 self.operationQueue[app.icon] = operationDown; } } return cell; } // 实现代理方法 -(void)operation:(XBOperation *)operation didFinishDownloadImage:(UIImage *)image { // 必须清楚当前operation 防止由于网络等原因造成的下载失败 而operation还在字典中 这样永远下载不了 [self.operationQueue removeObjectForKey:operation.url]; if (image) { // 将图片放在缓存字典中 self.imageCache[operation.url] = image; // 刷新表格 单行刷新 [self.tableView reloadRowsAtIndexPaths:@[operation.indexPath] withRowAnimation:UITableViewRowAnimationNone]; } }