iOS开发之UICollectionViewDataSourcePrefetching

简介: 在iOS10中,苹果为UICollectionViewCell引入了Pre-Fetching预加载机制用于提升它的性能。主要引入了一个新的数据源协议UICollectionViewDataSourcePrefetching,包含两个方法:@proto...

在iOS10中,苹果为UICollectionViewCell引入了Pre-Fetching预加载机制用于提升它的性能。主要引入了一个新的数据源协议UICollectionViewDataSourcePrefetching,包含两个方法:

@protocol UICollectionViewDataSourcePrefetching <NSObject>
@required
// 预加载数据
- (void)collectionView:(UICollectionView *)collectionView prefetchItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths NS_AVAILABLE_IOS(10_0);
@optional
// 取消提前加载数据
- (void)collectionView:(UICollectionView *)collectionView cancelPrefetchingForItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths  NS_AVAILABLE_IOS(10_0);
@end

网上搜了一大圈,讲述原理的(翻译文档)的文章很多,有干货的Demo很少,于是乎自己摸索了一下,写了一个简单的案例,在此记录并分享一下。

运行环境:Xcode 8.2.1 + iOS 10.2

核心步骤:

1、遵从 UICollectionViewDataSourcePrefetching 协议
2、实现 collectionView:prefetchItemsAtIndexPaths 方法和collectionView:cancelPrefetchItemsAtIndexPaths 方法(可选)
3、将第1步中遵从协议的类设置为 UICollectionView 的 prefetchDataSource 属性

实现

一、创建UICollectionViewFlowLayout

自己写一个类继承自UICollectionViewFlowLayout

@implementation MyCollectionViewFlowLayout

-(void)prepareLayout{

    self.minimumLineSpacing = 1;//垂直间距
    self.minimumInteritemSpacing = 0;//水平间距
    self.sectionInset = UIEdgeInsetsMake(0, 0, 8, 0);//分组间距

}
@end

二、用XIB定义一个

里面就一个UIImageView,然后拽线设置一个IBOutlet


img_2e9b38821af2c3d2186c5ed9b286b452.png
UICollectionViewCell.png
@property (weak, nonatomic) IBOutlet UIImageView *imgView;

三、控制器

注释很详细

#import "ViewController.h"
#import "MyCollectionViewFlowLayout.h"
#import "ImgCollectionViewCell.h"

#define ScreenW [UIScreen mainScreen].bounds.size.width
//重用标识
static NSString *cellId = @"imgCell";
//遵守协议
@interface ViewController ()<UICollectionViewDelegate, UICollectionViewDataSource, UICollectionViewDataSourcePrefetching>

//下载图片任务
@property(nonatomic, strong) NSMutableDictionary<NSURL *, dispatch_queue_t> *tasks;
//图片地址
@property(nonatomic, copy) NSMutableArray<NSURL *> *imgURLArray;
//下载的图片
@property(nonatomic, copy) NSMutableDictionary<NSURL *, UIImage *> *imgs;
//UICollectionView
@property(nonatomic, weak) UICollectionView *collectionView;

@end

@implementation ViewController

//懒加载imgURLArray
-(NSMutableArray<NSURL *> *)imgURLArray{
    
    if (_imgURLArray == nil) {   
        _imgURLArray = [NSMutableArray array];    
        for (int i = 0; i < 30; i++) {
            NSURL *imgURL = [NSURL URLWithString:@"https://timgsa.baidu.com/timg?image&quality=80&size=b9999_10000&sec=1494499175005&di=1d8d40ac84f4a71cb26d7bf4a5a845ec&imgtype=0&src=http%3A%2F%2Fimg10.360buyimg.com%2Fyixun_zdm%2Fjfs%2Ft2830%2F11%2F2310606472%2F165925%2F962fa94a%2F575f7664Nfd743845.jpg"];
            [_imgURLArray addObject:imgURL];            
        }
    }
    
    return _imgURLArray;    
}

//懒加载imgs
-(NSMutableDictionary<NSURL *,UIImage *> *)imgs{
    if (_imgs == nil) {        
        _imgs = [NSMutableDictionary dictionary];
    }   
    return _imgs;
}

//懒加载tasks
-(NSMutableDictionary<NSURL *,dispatch_queue_t> *)tasks{
    if (_tasks == nil) {       
        _tasks = [NSMutableDictionary dictionary];
    }   
    return _tasks;   
}


- (void)viewDidLoad {
    [super viewDidLoad];
    
    //创建UICollectionView
    //创建布局
    UICollectionViewLayout *layout = [[MyCollectionViewFlowLayout alloc]init];
    //初始化一个UICollectionView
    UICollectionView *collection = [[UICollectionView alloc]initWithFrame:[UIScreen mainScreen].bounds collectionViewLayout:layout];
    //设置背景色
    collection.backgroundColor = [UIColor groupTableViewBackgroundColor];
    //设置代理
    collection.dataSource = self;
    collection.delegate = self;
    collection.prefetchDataSource = self;
    //注册Cell
    UINib *nib = [UINib nibWithNibName:@"ImgCollectionViewCell" bundle:nil];
    [collection registerNib:nib forCellWithReuseIdentifier:cellId];
     
    [self.view addSubview:collection];
    
    self.collectionView = collection;
    
}


-(void)loadImage:(NSIndexPath *)indexPath{
    
    NSURL *currentURL = [self.imgURLArray objectAtIndex:indexPath.row];
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);  
    __weak typeof(self) weakSelf = self;
    
    //异步下载图片
    dispatch_async(queue, ^{
      
        NSData *imageData = [NSData dataWithContentsOfURL:currentURL];
        UIImage *image = [UIImage imageWithData:imageData];
        weakSelf.imgs[currentURL] = image;
        
        //更新UI   
         dispatch_async(dispatch_get_main_queue(), ^{
            
            ImgCollectionViewCell *cell = (ImgCollectionViewCell *)[weakSelf.collectionView cellForItemAtIndexPath:indexPath];
            cell.imgView.image = image;
        });
    });
    
    //为了取消任务
    self.tasks[currentURL] = queue;
    
}

#pragma mark <UICollectionViewDataSource>
- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section{
    
    return self.imgURLArray.count;
}

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath{
     
    ImgCollectionViewCell * cell = [collectionView dequeueReusableCellWithReuseIdentifier:cellId forIndexPath:indexPath];
    
    // 获取URL
    NSURL *imgURL = self.imgURLArray[indexPath.row];
    
    //对应的URL的图片已经存在
    if (self.imgs[imgURL]) {        
        cell.imgView.image = self.imgs[imgURL];
    }
    //不存在
    else{
        
        [self loadImage:indexPath];
    }
    
    return cell;
    
}


#pragma mark <UICollectionViewDelegate>
//定义每个Item 的大小
- (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout*)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath *)indexPath { 
    CGFloat W = (ScreenW - 1) / 2;    
    return CGSizeMake(W, 100);
    
}

#pragma mark <UICollectionViewDataSourcePrefetching>
- (void)collectionView:(UICollectionView *)collectionView prefetchItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths {
    
    for (NSIndexPath * indexPath in indexPaths) {
        
        NSURL *currentURL = [self.imgURLArray objectAtIndex:indexPath.row]; 
        //不存在就请求
        if (!self.imgs[currentURL]) {
            [self loadImage:indexPath];
        }
    }
    
}


- (void)collectionView:(UICollectionView *)collectionView cancelPrefetchingForItemsAtIndexPaths:(NSArray<NSIndexPath *> *)indexPaths {
    
    for (NSIndexPath * indexPath in indexPaths){
        
        NSURL *currentURL = [self.imgURLArray objectAtIndex:indexPath.row];  
        //当前任务存在
        if (self.tasks[currentURL]) {
            dispatch_queue_t queue = self.tasks[currentURL];
            dispatch_suspend(queue);
            self.tasks[currentURL] = nil;
        }
    }
    
}

@end

效果

img_2708c27e640a433c1b67450826a998f9.gif
效果演示.gif

写在后面的话

1、这个新特性仍然需要探究
2、遇到的一个坑:细心看的话可以发现我的字典是懒加载的,如果直接在viewDidLoad中初始化会在 weakSelf.imgs[currentURL] = image; 一行报错,why?烦请知道的告知。

源代码

目录
相关文章
|
27天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
4天前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
85 66
|
14天前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
|
18天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
20天前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
23天前
|
安全 IDE Swift
探索iOS开发之旅:从初学者到专家
在这篇文章中,我们将一起踏上iOS开发的旅程,从基础概念的理解到深入掌握核心技术。无论你是编程新手还是希望提升技能的开发者,这里都有你需要的指南和启示。我们将通过实际案例和代码示例,展示如何构建一个功能齐全的iOS应用。准备好了吗?让我们一起开始吧!
|
28天前
|
安全 Swift iOS开发
Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法
本文深入探讨了 Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法。Swift 以其简洁、高效和类型安全的特点,结合 UIKit 丰富的组件和功能,为开发者提供了强大的工具。文章从 Swift 的语法优势、类型安全、编程模型以及与 UIKit 的集成,到 UIKit 的主要组件和功能,再到构建界面的实践技巧和实际案例分析,全面介绍了如何利用这些技术创建高质量的用户界面。
29 2
|
1月前
|
安全 数据处理 Swift
深入探索iOS开发中的Swift语言特性
本文旨在为开发者提供对Swift语言在iOS平台开发的深度理解,涵盖从基础语法到高级特性的全面分析。通过具体案例和代码示例,揭示Swift如何简化编程过程、提高代码效率,并促进iOS应用的创新。文章不仅适合初学者作为入门指南,也适合有经验的开发者深化对Swift语言的认识。
48 9
|
1月前
|
vr&ar Android开发 iOS开发
安卓与iOS开发中的用户界面设计原则
【10月更文挑战第41天】探索移动应用开发的精髓,本文将深入分析安卓和iOS平台上用户界面设计的核心原则。通过比较两大操作系统的设计哲学,我们将揭示如何打造直观、易用且美观的应用程序界面。无论你是初学者还是资深开发者,这篇文章都将为你提供宝贵的见解和实用的技巧,帮助你在竞争激烈的应用市场中脱颖而出。
|
1月前
|
设计模式 Swift iOS开发
探索iOS开发:从基础到高级,打造你的第一款App
【10月更文挑战第40天】在这个数字时代,掌握移动应用开发已成为许多技术爱好者的梦想。本文将带你走进iOS开发的世界,从最基础的概念出发,逐步深入到高级功能实现,最终指导你完成自己的第一款App。无论你是编程新手还是有志于扩展技能的开发者,这篇文章都将为你提供一条清晰的学习路径。让我们一起开始这段旅程吧!