ios开发多线程四:nsoperation多图下载综合案例

Hello_IOS Hello_IOS     2022-08-03     451

关键词:

#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 = 5;
    }
    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 1;
}

-(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多线程开发系列(一)三种多线程办法进行对比性能和操作的复杂度,并认真介绍了NSThread的使用。我们借助上一次的例子完全可以采取NSOperation方法进行实现NSOperation不具备封装操作的能力,必须使用... 查看详情

ios开发多线程编程2-nsoperation

1.简介NSOperation实例封装了需要执行的操作和执行操作所需的数据,并且能够以并发或非并发的方式执行这个操作。NSOperation本身是抽象基类,因此必须使用它的子类,使用NSOperation子类的方式有2种:1>Foundation框架提供了两个具... 查看详情

ios开发多线程篇11—自定义nsoperation

     iOS开发多线程篇—自定义NSOperation一、实现一个简单的tableView显示效果实现效果展示: 代码示例(使用以前在主控制器中进行业务处理的方式)1.新建一个项目,让控制器继承自UITableViewController。1//2... 查看详情

ios开发多线程篇10—nsoperation基本操作

       iOS开发多线程篇—NSOperation基本操作一、并发数(1)并发数:同时执?行的任务数.比如,同时开3个线程执行3个任务,并发数就是3(2)最大并发数:同一时间最多只能执行的任务的个数。(3)最?大... 查看详情

ios开发多线程一:了解-nsoperation的基本使用

 #import"ViewController.h"@interfaceViewController()@end@implementationViewController-(void)touchesBegan:(NSSet<UITouch*>*)toucheswithEvent:(UIEvent*)event{[selfblockOperation];}/***1:NSOpe 查看详情

ios多线程编程(四)------gcd(grandcentraldispatch)

一、简介是基于C语言开发的一套多线程开发机制,也是目前苹果官方推荐的多线程开发方法,用起来也最简单,只是它基于C语言开发,并不像NSOperation是面向对象的开发,而是完全面向过程的。如果使用GCD... 查看详情

多线程异步加载图片

多图片多线程异步下载开发中非常常用的就是就是图片下载,我们常用的就是SDWebImage,但是作为开发人员,不仅要能会用,还要知道其原理。本文就会介绍多图下载的实现。本文中的示例Demno地址,下载后项目位于iOS_Demo/09-多图... 查看详情

[ios开发]nsoperation&nsoperationqueue(代码片段)

NSOperation、NSOperationQueue是苹果提供给我们的一套多线程解决方案。是基于GCD更高一层的封装,完全面向对象。但是比GCD更简单易用、代码可读性也更高。(一)使用步骤NSOperation实现多线程的使用步骤分为三步:... 查看详情

[ios开发]nsoperation&nsoperationqueue(代码片段)

NSOperation、NSOperationQueue是苹果提供给我们的一套多线程解决方案。是基于GCD更高一层的封装,完全面向对象。但是比GCD更简单易用、代码可读性也更高。(一)使用步骤NSOperation实现多线程的使用步骤分为三步:... 查看详情

ios多线程编程--nsoperation(代码片段)

...,我们需要实例化NSOperation的具体子类来进行多线程开发。3.GrandCentralDispatch:简称GCD,iOS4才开始支持.提供了一些新特性、运行库来支持多核并行编程,它的关注点更高:如何在多个cpu上提升效率了解GCD点这里... 查看详情

ios多线程总结——nsoperation与nsoperationqueue的使用

本篇是多线程总结的第三篇,关于多线程的概念和NSThread的使用写在第一篇,《iOS多线程总结(1)——多线程相关概念及NSObject/NSThread的使用》,第二篇《iOS多线程总结(2)——GCD》主要讲解GCD的使用,本编是线程实现... 查看详情

ios多线程——你要知道的nsoperation都在这里(代码片段)

你要知道的iOS多线程NSThread、GCD、NSOperation、RunLoop都在这里转载请注明出处http://blog.csdn.net/u014205968/article/details/78323182本系列文章主要讲解iOS中多线程的使用,包括:NSThread、GCD、NSOperation以及RunLoop的使用方法详解,本系列... 查看详情

多线程之pthread,nsthread,nsoperation,gcd

...命周期需要程序员自己管理,使用难度较大,所以在实际开发中通常不使用。NThread:是苹果对pthread的一个 查看详情

ios开发多线程详解

常用的多线程开发有三种方式:1.NSThread2.NSOperation3.GCD线程状态分为isExecuting(正在执行)、isFinished(已经完成)、isCancellled(已经取消)三种。其中取消状态程序可以干预设置,只要调用线程的canc... 查看详情

ios多线程之nsoperation(代码片段)

IOS多线程之NSOperation(2)最大并发数openvarmaxConcurrentOperationCount:Int并发数就是同时执行的任务数。比如,同时开3个线程执行3个任务,并发数就是3。但是,并发数是3,并不代表开启的线程数就是3,也有可能是4个或者5个。因为线程... 查看详情

巧谈gcd

...中,苹果推荐也是我们最经常使用的无疑是GCD。对于身为开发者的我们来说,并发一直都很棘手,如果对GCD的理解不够透彻,那么iOS开发的历程绝对不会顺利。这里,我会从几个角度浅谈我对GCD的理解。一、多线程背景Althoughthre... 查看详情

ios多线程之nsoperation(代码片段)

IOS多线程之NSOperation(3)操作优先级和服务质量可以通过QueuePriority属性来设置operation在队列中的执行优先级publicenumQueuePriority:Int,@uncheckedSendablecaseveryLow=-8caselow=-4casenormal=0casehigh=4caseveryHigh=8IOS8.0之后推荐使用QualityOfService来设置operati... 查看详情

ios多线程编程——gcd与nsoperation总结

...来,我个人(可能还有很多同学),对多线程编程都存在一些误解。一个很明显的表现是,很多人有这样的看法:新开一个线程,能提高速度,避免阻塞主线程毕竟多线程嘛,几个线程一起跑任... 查看详情