lazy懒加载(延迟加载)UITableView

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

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

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

@下面我的代码用的是自己写的异步加载缓存类,SDWebImage的加载图片的懒加载,会在后面的章节给出.(为什么不同呢,因为SDWebImage我以前使用重来不关心它将图片存储在沙盒中的名字和路径,但是要实现懒加载的话,一定要得到图片路径,所以在找SDWebImage如何存储图片路径上花了点时间)

@model类
#import <Foundation/Foundation.h>

@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 <Foundation/Foundation.h>

@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];

}

lazy懒加载(延迟加载)UITableView

时间: 2024-10-09 18:26:15

lazy懒加载(延迟加载)UITableView的相关文章

tableview加载图片的时候的优化之lazy(懒加载)模式and异步加载模式

iOS---tableview加载图片的时候的优化之lazy(懒加载)模式and异步加载模式举个例子,当我们在用网易新闻App时,看着那么多的新闻,并不是所有的都是我们感兴趣的,有的时候我们只是很快的滑过,想要快速的略过不喜欢的内容,但是只要滑动经过了,图片就开始加载了,这样用户体验就不太好,而且浪费内存.这个时候,我们就可以利用lazy加载技术,当界面滑动或者滑动减速的时候,都不进行图片加载,只有当用户不再滑动并且减速效果停止的时候,才进行加载.刚开始我异步加载图片利用SDWebImage来做

解决hibernate中的懒加载(延迟加载)问题

解决hibernate中的懒加载(延迟加载)问题 我们在开发的时候经常会遇到延迟加载问题,在实体映射时,多对一和多对多中,多的一样的属性默认是lazy="true"(即,默认是延迟加载), 如:<many-to-one name="parent" class="Department" column="parentId" lazy="true"/> 延迟加载表现在:比如:我们要查询id为2的部门数

020 &lt;one-to-one&gt;、&lt;many-to-one&gt;单端关联上的lazy(懒加载)属性

<one-to-one>.<many-to-one>单端关联上,可以取值:false/proxy/noproxy(false/代理/不代理) 实例一:所有lazy属性默认(支持懒加载) session = HibernateUtils.getSession(); tx = session.beginTransaction(); //不发出SQL语句,支持lazy(懒加载) User user = (User) session.load(User.class, 3); //发出SQL

018 关联映射文件中&lt;class&gt;标签中的lazy(懒加载)属性

Lazy(懒加载): 只有在正真使用该对象时,才会创建这个对象 Hibernate中的lazy(懒加载): 只有我们在正真使用时,它才会发出SQL语句,给我们去查询,如果不使用对象则不会发SQL语句进行查询. Hibernate中lazy(懒加载)的实现: 采用了第三方组件的库,这个库叫cglib.jar(比较流行),这个库对我们的类生成代理类(JDK的动态代理,只能对JDK中实现了接口的类进行代理),代理可以控制源对象并且可以对源对象的功能进行增强,而cglib.jar可以对类进行代理(cgl

iOS---tableview加载图片的时候的优化之lazy(懒加载)模式and异步加载模式

举个例子,当我们在用网易新闻App时,看着那么多的新闻,并不是所有的都是我们感兴趣的,有的时候我们只是很快的滑过,想要快速的略过不喜欢的内容,但是只要滑动经过了,图片就开始加载了,这样用户体验就不太好,而且浪费内存. 这个时候,我们就可以利用lazy加载技术,当界面滑动或者滑动减速的时候,都不进行图片加载,只有当用户不再滑动并且减速效果停止的时候,才进行加载. 刚开始我异步加载图片利用SDWebImage来做,最后试验的时候出现了重用bug,因为虽然SDWebImage实现了异步加载缓存,当加载

懒加载(延迟加载)之后,在使用数据过程中容易出现的bug

在UI中,使用懒加载,也就是延迟加载来加载数据的时候,总是会面临几个问题? 如:1. >为什么先创建NSArray属性? 2. >为什么重写NSArray的get方法? 3.>为什么要判断是否为空? 4.>为什么下方代码"//1"这里不用NSString stringWithFormat: 而"//2"这里要使用? 5.>同时"//2"这里为什么使用的是%ld 来作为占位符? 这些问题不搞懂,懒加载就很难通透,  代

019 关联映射文件中集合标签中的lazy(懒加载)属性

<set>.<list>集合上,可以取值:true/false/extra,(默认值为:true) 实例一:(集合上的lazy=true(默认))class默认lazy=true(默认) session = HibernateUtils.getSession(); tx = session.beginTransaction(); //不会发出SQL语句 Classes classes = (Classes)session.load(Classes.class, 1); //发出SQ

hibernate懒加载

16.懒加载 性能: 发出的sql语句越少,性能越高 方法: 1.懒加载 2.抓取策略 3.缓存策略 4.HQL语句   懒加载 1.类的懒加载 1.利用session.load方法可以产生代理对象 2.在session.load方法执行的时候并不发出sql语句 3.在得到其一般属性的时候发出sql语句 4.只针对一般属性有效,针对标示符属性是无效的 5.默认情况就是懒加载  2.集合的懒加载 <set name=""  lazy="" cascasd=&qu

smartjs - DataManager 场景示例分析 - 数据懒加载

发一张policy的参数图设置图: 场景1 - 数据的懒加载/延迟加载 在很多时候,为了提高网页的加载速度,减少不必要的开销,会将页面的数据拆分成几个部分,首先加载呈现可视区域内的数据,然后剩下来的会在需要的时候在进行加载. 而这种按需加载的数据又分为两种: 1.按照需要进行加载:可以是由某个动作触发来引起,比如:tab,查看更多等: 2.采用缓存的方式:对后续动作的预知,提前将后续的数据加载进来,放入到缓存中:等需要的时候能提供快速的响应:比如:很多igrid的滚动分页 那么来看一下在data