iOS多线程自定义operation加载图片 不重复下载图片

时间:2022-02-21 14:56:10

摘要: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文件

 #import <Foundation/Foundation.h>
@class XBOperation; @protocol XBOperationDelegate <NSObject> -(void)operation:(XBOperation *)operation didFinishDownloadImage:(UIImage *)image; @end
/**
* 自定义operation
*/
@interface XBOperation : NSOperation // 代理
@property (nonatomic, weak)id<XBOperationDelegate> delegate; /**
* url
*/
@property (nonatomic, copy) NSString *url;
/**
* indexPath 和tableview中的cell相对应
*/
@property (nonatomic, strong) NSIndexPath *indexPath; @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 ;
} -(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];
}
}