说到懒加载,其实就是延迟会再加载,没有想象中的那么神秘,其实我们时刻都在用懒加载,

第一种:简单的延迟创建控件,比如说,创建一个属性变量,我们用get方法来获取生成这个变量就是用到了懒加载,详细点说就是,又一个label属性变量,我们如果只直接初始化的时候创建它,当然可以,但是会消耗内存,一个view中有一个,两个可以,如果有几十个呢,会不会就有明显的效果呢,所以这个时候我们会用到懒加载,代码如下:

- (UILabel *)nameLabel {

if (!_nameLabel) {
        _nameLabel = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 20, 20)];
    }
    return _nameLabel;

}

就是get方法,先判断是否有,没有创建,那个地方用的时候,我直接用self.nameLabel来调用这个方法即可。

第二种强大的优化: table在加载图片的时候的优化

比如说有这么个需求,当我们在用网易新闻App时,看着那么多的新闻,并不是所有的都是我们感兴趣的,有的时候我们只是很快的滑过,想要快速的略过不喜欢的内容,但是只要滑动经过了,图片就开始加载了,这样用户体验就不太好,而且浪费内存.

这个时候,我们就可以利用lazy加载技术,当界面滑动或者滑动减速的时候,都不进行图片加载,只有当用户不再滑动并且减速效果停止的时候,才进行加载.

刚开始我异步加载图片利用SDWebImage来做,最后试验的时候出现了重用bug,因为虽然SDWebImage实现了异步加载缓存,当加载完图片后再请求会直接加载缓存中的图片,注意注意注意,关键的来了,如果是lazy加载,滑动过程中是不进行网络请求的,cell上的图片就会发生重用,当你停下来能进行网络请求的时候,才会变回到当前Cell应有的图片,大概1-2秒的延迟吧(不算延迟,就是没有进行请求,也不是没有缓存的问题).怎么解决呢?这个时候我们就要在Model对象中定义个一个UIImage的属性,异步下载图片后,用已经缓存在沙盒中的图片路径给它赋值,这样,才cellForRowAtIndexPath方法中,判断这个UIImage对象是否为空,若为空,就进行网络请求,不为空,就直接将它赋值给cell的imageView对象,这样就能很好的解决图片短暂重用问题.

@下面我的代码用的是自己写的异步加载缓存类,SDWebImage的加载图片的懒加载,原理差不多.

@model类
#import @interface NewsItem : NSObject@property (nonatomic,copy) NSString * newsTitle;
@property (nonatomic,copy) NSString * newsPicUrl;
@property (nonatomic,retain) UIImage * newsPic; //  存储每个新闻自己的image对象- (id)initWithDictionary:(NSDictionary *)dic;//  处理解析
+ (NSMutableArray *)handleData:(NSData *)data;
@end#import "NewsItem.h"
#import "ImageDownloader.h"@implementation NewsItem- (void)dealloc
{self.newsTitle = nil;self.newsPicUrl = nil;self.newsPic = nil;[super dealloc];
}- (id)initWithDictionary:(NSDictionary *)dic
{self = [super init];if (self) {self.newsTitle = [dic objectForKey:@"title"];self.newsPicUrl = [dic objectForKey:@"picUrl"];//从本地沙盒加载图像ImageDownloader * downloader = [[[ImageDownloader alloc] init] autorelease];self.newsPic = [downloader loadLocalImage:_newsPicUrl];}return self;
}+ (NSMutableArray *)handleData:(NSData *)data;
{//解析数据NSError * error = nil;NSDictionary * dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:&error];NSMutableArray * originalArray = [dic objectForKey:@"news"];//封装数据对象NSMutableArray * resultArray = [NSMutableArray array];for (int i=0 ;i<[originalArray count]; i++) {NSDictionary * newsDic = [originalArray objectAtIndex:i];NewsItem * item = [[NewsItem alloc] initWithDictionary:newsDic];[resultArray addObject:item];[item release];}return resultArray;}@end
@图片下载类
#import @class NewsItem;@interface ImageDownloader : NSObject@property (nonatomic,copy) NSString * imageUrl;
@property (nonatomic,retain) NewsItem * newsItem; //下载图像所属的新闻//图像下载完成后,使用block实现回调
@property (nonatomic,copy) void (^completionHandler)(void);//开始下载图像
- (void)startDownloadImage:(NSString *)imageUrl;//从本地加载图像
- (UIImage *)loadLocalImage:(NSString *)imageUrl;@end#import "ImageDownloader.h"
#import "NewsItem.h"@implementation ImageDownloader- (void)dealloc
{self.imageUrl = nil;Block_release(_completionHandler);[super dealloc];
}#pragma mark - 异步加载
- (void)startDownloadImage:(NSString *)imageUrl
{self.imageUrl = imageUrl;// 先判断本地沙盒是否已经存在图像,存在直接获取,不存在再下载,下载后保存// 存在沙盒的Caches的子文件夹DownloadImages中UIImage * image = [self loadLocalImage:imageUrl];if (image == nil) {// 沙盒中没有,下载// 异步下载,分配在程序进程缺省产生的并发队列dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// 多线程中下载图像NSData * imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:imageUrl]];// 缓存图片[imageData writeToFile:[self imageFilePath:imageUrl] atomically:YES];// 回到主线程完成UI设置dispatch_async(dispatch_get_main_queue(), ^{//将下载的图像,存入newsItem对象中UIImage * image = [UIImage imageWithData:imageData];self.newsItem.newsPic = image;//使用block实现回调,通知图像下载完成if (_completionHandler) {_completionHandler();}});});}}#pragma mark - 加载本地图像
- (UIImage *)loadLocalImage:(NSString *)imageUrl
{self.imageUrl = imageUrl;// 获取图像路径NSString * filePath = [self imageFilePath:self.imageUrl];UIImage * image = [UIImage imageWithContentsOfFile:filePath];if (image != nil) {return image;}return nil;
}#pragma mark - 获取图像路径
- (NSString *)imageFilePath:(NSString *)imageUrl
{// 获取caches文件夹路径NSString * cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];// 创建DownloadImages文件夹NSString * downloadImagesPath = [cachesPath stringByAppendingPathComponent:@"DownloadImages"];NSFileManager * fileManager = [NSFileManager defaultManager];if (![fileManager fileExistsAtPath:downloadImagesPath]) {[fileManager createDirectoryAtPath:downloadImagesPath withIntermediateDirectories:YES attributes:nil error:nil];}#pragma mark 拼接图像文件在沙盒中的路径,因为图像URL有"/",要在存入前替换掉,随意用"_"代替NSString * imageName = [imageUrl stringByReplacingOccurrencesOfString:@"/" withString:@"_"];NSString * imageFilePath = [downloadImagesPath stringByAppendingPathComponent:imageName];return imageFilePath;
}@end
@这里只给出关键代码,网络请求,数据处理,自定义cell自行解决#pragma mark - Table view data source- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{// Return the number of sections.return 1;
}- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{// Return the number of rows in the section.if (_dataArray.count == 0) {return 10;}return [_dataArray count];
}- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{static NSString *cellIdentifier = @"Cell";NewsListCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier ];if (!cell) {cell = [[[NewsListCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier] autorelease];}NewsItem * item = [_dataArray objectAtIndex:indexPath.row];cell.titleLabel.text = item.newsTitle;//判断将要展示的新闻有无图像if (item.newsPic == nil) {//没有图像下载cell.picImageView.image = nil;NSLog(@"dragging = %d,decelerating = %d",self.tableView.dragging,self.tableView.decelerating);// ??执行的时机与次数问题if (self.tableView.dragging == NO && self.tableView.decelerating == NO) {[self startPicDownload:item forIndexPath:indexPath];}}else{//有图像直接展示NSLog(@"1111");cell.picImageView.image = item.newsPic;}cell.titleLabel.text = [NSString stringWithFormat:@"indexPath.row = %ld",indexPath.row];return cell;
}- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{return [NewsListCell cellHeight];
}//开始下载图像
- (void)startPicDownload:(NewsItem *)item forIndexPath:(NSIndexPath *)indexPath
{//创建图像下载器ImageDownloader * downloader = [[ImageDownloader alloc] init];//下载器要下载哪个新闻的图像,下载完成后,新闻保存图像downloader.newsItem = item;//传入下载完成后的回调函数[downloader setCompletionHandler:^{//下载完成后要执行的回调部分,block的实现//根据indexPath获取cell对象,并加载图像
#pragma mark cellForRowAtIndexPath-->没看到过NewsListCell * cell = (NewsListCell *)[self.tableView cellForRowAtIndexPath:indexPath];cell.picImageView.image = downloader.newsItem.newsPic;}];//开始下载[downloader startDownloadImage:item.newsPicUrl];[downloader release];
}- (void)loadImagesForOnscreenRows
{
#pragma mark indexPathsForVisibleRows-->没看到过//获取tableview正在window上显示的cell,加载这些cell上图像。通过indexPath可以获取该行上需要展示的cell对象NSArray * visibleCells = [self.tableView indexPathsForVisibleRows];for (NSIndexPath * indexPath in visibleCells) {NewsItem * item = [_dataArray objectAtIndex:indexPath.row];if (item.newsPic == nil) {//如果新闻还没有下载图像,开始下载[self startPicDownload:item forIndexPath:indexPath];}}
}#pragma mark - 延迟加载关键
//tableView停止拖拽,停止滚动
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate
{//如果tableview停止滚动,开始加载图像if (!decelerate) {[self loadImagesForOnscreenRows];}NSLog(@"%s__%d__|%d",__FUNCTION__,__LINE__,decelerate);
}- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView
{//如果tableview停止滚动,开始加载图像[self loadImagesForOnscreenRows];}
Hope To Help You !

技术交流群:141624834 进群请说你看的那篇博客,我们一起探讨成长

 

同类文章:

通信机制之异步通信和同步通信

iOS 同步 异步 并发 并行 多线程的讲解与区别

Hope To Help You !

技术交流群:141624834 进群请说你看的那篇博客,我们一起探讨成长

lazy(懒加载)模式和异步加载模式详解相关推荐

  1. 异步加载场景与异步加载游戏资源

    异步加载场景与异步加载游戏资源 Application.LoadLevel 加载关卡 在同步加载游戏场景的时候通常会使用方法Application.LoadLevel 加载关卡. 该方法在执行完App ...

  2. Lesson13【加餐】 损失函数的随机创建现象详解

    [Lesson 13 加餐]损失函数的随机创建现象详解   接下来,我们通过手动创建一个实例,来观察在小批梯度下降过程中,损失函数是如何根据数据数据变化而变化的,这里既是作为本节内容的一个补充,同时也 ...

  3. 【5G/4G】加/解密+完整性保护/校验算法源码详解

    文章目录 加/解密+完整性保护/校验算法源码详解 一.加解密算法 二.完整性保护/校验算法 本人就职于国际知名终端厂商,负责modem芯片研发. 在5G早期负责终端数据业务层.核心网相关的开发工作,目 ...

  4. 命令行模式下几个网络命令详解

    命令行模式下几个网络命令详解 上一篇 / 下一篇  2007-05-16 16:20:55 查看( 166 ) / 评论( 0 ) / 评分( 0 / 0 ) 命令行模式下几个网络命令详解 一.pin ...

  5. WEB后台--邮件和短信业务实现(包括Java一键实现、封装和异步)以及原理详解

    本来就打算针对一些固定的特别点的业务(QQ与网易邮件.拦截设计.短信.定时器等等)来进行记录以及解析原理,这些会比较零散记录在JavaWeb的分类里面,感兴趣的童鞋可以去看下. 有人问为什么要邮件短信 ...

  6. 工程之星android版使用,安卓版工程之星软件网络1+1模式及网络cors连接操作详解...

    原标题:安卓版工程之星软件网络1+1模式及网络cors连接操作详解 现在,越来越多用户开始使用安卓版工程之星进行作业,科力达技术工程师总结了安卓版工程之星网络1+1模式及网络CORS连接方式操作步骤, ...

  7. python协程详解_对Python协程之异步同步的区别详解

    一下代码通过协程.多线程.多进程的方式,运行代码展示异步与同步的区别. import gevent import threading import multiprocessing # 这里展示同步和异 ...

  8. java 配置jmstemplate_SpringBoot集成JmsTemplate(队列模式和主题模式)及xml和JavaConfig配置详解...

    1.导入jar包: org.springframework.boot spring-boot-starter-activemq org.apache.activemq activemq-pool 2. ...

  9. java ajax同步请求,成都汇智动力-java ajax实现异步同步请求全面详解

    原标题:成都汇智动力-java ajax实现异步同步请求全面详解 对象 var request=new () //兼容IE5 IE6 if (window.) {// code for IE7+, F ...

最新文章

  1. Ptyhon学习之元组
  2. bzoj3223 splay
  3. css学习_cs3s旋转的图片
  4. es6 dsl与sql对比
  5. KVO 实现两个页面之间的通信
  6. vim linux python3,VIM:在python-mode中使用python3解释器
  7. 车牌识别程序python_车牌检测和识别的Python应用软件实现
  8. 全网首发:FFMpeg使用NVIDIA DECODER,解码后的数据转换为YUV420P
  9. 数据库系统概论第五版课后习题答案王珊
  10. 1-docker安装
  11. c语言precede函数怎么构造,数据结构——栈的应用(表达式求值)(C语言)
  12. 百度有啊创始团队写博客 自曝高层动荡内幕
  13. 计算机U盘驱动程序安装不上问题解决方案
  14. 人才缺口30万,市场需求每年涨20% ,这一 IT 岗位你知道嘛
  15. ngx_http_core_module模块提供的变量
  16. Python爬虫 --- Scrapy爬取IT桔子网
  17. 苹果消息推送服务教程:第一部分(共2部分)
  18. Ubuntu 20.04没有声音播放时出现哒哒的噪音
  19. linux文件系统层级结构标准
  20. ORACLE压力测试工具orion

热门文章

  1. php文件直链源码,蓝奏网盘文件夹直链解析源码
  2. android vitamio框架,利用Vitamio框架来做自己的视频播放器
  3. DCC-MGARCH:动态条件相关系数模型(R+Stata)
  4. 涉密计算机系统分级保护,什么是分级保护!!
  5. 英语作文计算机国际会议开幕词,【国际会议开幕式】国际会议开幕词英文
  6. 单电源运放滤波器设计
  7. C语言游戏之贪吃蛇--链表实现
  8. Spyder 错误:error: (-215:Assertion failed) dst.data = src.data in function ‘cv::undistort‘
  9. 荣耀平板5鸿蒙降级安卓并刷入原生Android12系统——麒麟659,4+64G,10英寸wifi版本
  10. 毕业设计 SSM毕业设计管理系统