zoukankan      html  css  js  c++  java
  • 多线程实现多图片下载2

    最终的演示如下

    这次是用多线程进行图片的下载与存储,而且考虑到下载失败,占位图片的问题(第一张就是下载失败的图片)

    闲话少说,上代码吧,因为有一部分和上次的一样,所以这里只上传不一样的

    依旧都是在ViewController.m中

    1.

    @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.

    - (NSOperationQueue *)queue {
        if (!_queue) {
            _queue = [[NSOperationQueue alloc] init];
            //最大并发数
            _queue.maxConcurrentOperationCount = 3;
        }
        return _queue;
    }

    对queue的初始化,以及控制子线程最多为3条

    3.

    - (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;
    }
    View Code

    这次绑定数据的方法内容有点多,因为考虑到了不少细节,不过逻辑和上次的差不多。

    下次再画一个详细的流程图~

  • 相关阅读:
    jquery---stop()
    标准设备的-media queries
    CSS3 Media Queries
    css---------selector
    sublime----(1)
    XAMPP、WAMP、appserv
    Express.js----路由
    nodejs----上传显示图片
    node.js-----处理POST请求
    DotNetOpenAuth使用笔记
  • 原文地址:https://www.cnblogs.com/langji/p/5328497.html
Copyright © 2011-2022 走看看