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

iOS---tableview加载图片的时候的优化之lazy(懒加载)模式and异步加载模式举个例子,当我们在用网易新闻App时,看着那么多的新闻,并不是所有的都是我们感兴趣的,有的时候我们只是很快的滑过,想要快速的略过不喜欢的内容,但是只要滑动经过了,图片就开始加载了,这样用户体验就不太好,而且浪费内存.这个时候,我们就可以利用lazy加载技术,当界面滑动或者滑动减速的时候,都不进行图片加载,只有当用户不再滑动并且减速效果停止的时候,才进行加载.刚开始我异步加载图片利用SDWebImage来做,最后试验的时候出现了重用bug,因为虽然SDWebImage实现了异步加载缓存,当加载完图片后再请求会直接加载缓存中的图片,注意注意注意,关键的来了,如果是lazy加载,滑动过程中是不进行网络请求的,cell上的图片就会发生重用,当你停下来能进行网络请求的时候,才会变回到当前Cell应有的图片,大概1-2秒的延迟吧(不算延迟,就是没有进行请求,也不是没有缓存的问题).怎么解决呢?这个时候我们就要在Model对象中定义个一个UIImage的属性,异步下载图片后,用已经缓存在沙盒中的图片路径给它赋值,这样,才cellForRowAtIndexPath方法中,判断这个UIImage对象是否为空,若为空,就进行网络请求,不为空,就直接将它赋值给cell的imageView对象,这样就能很好的解决图片短暂重用问题.
@下面我的代码用的是自己写的异步加载缓存类,SDWebImage的加载图片的懒加载,原理差不多@model类#import <foundation foundation.h="">@interface NewsItem : [email protected] (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</foundation>@图片下载类#import <foundation foundation.h="">@class NewsItem;@interface ImageDownloader : [email protected] (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</foundation>@这里只给出关键代码,网络请求,数据处理,自定义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];}

时间: 2024-12-15 01:35:11

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

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

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

又优化了一下 Android ListView 异步加载图片

写这篇文章并不是教大家怎么样用listview异步加载图片,因为这样的文章在网上已经有很多了,比如这位仁兄写的就很好: http://www.iteye.com/topic/685986 我也是因为看了这篇文章而受到了启发. 先说说这篇文章的优点把,开启线程异步加载图片,然后刷新UI显示图片,而且通过弱引用缓存网络加载的图片,节省了再次连接网络的开销. 这样做无疑是非常可取的方法,但是加载图片时仍然会感觉到轻微的卡屏现象,特别是listview里的item在进行快速滑动的时候. 我找了一下原因,

页面优化——js异步加载

同步加载 在介绍js异步加载之前,我们先来看看什么是js同步加载.我们平时最常使用的就是这种同步加载形式: <script src="http://XXX.com/script.js"></script> 同步模式,又称阻塞模式,会阻止浏览器的后续处理,停止了后续的解析,因此停止了后续的文件加载(如图像).渲染.代码执行.一般的script标签(不带async等属性)加载时会阻塞浏览器,也就是说,浏览器在下载或执行该js代码块时,后面的标签不会被解析,例如在he

Android批量图片加载经典系列——采用二级缓存、异步加载网络图片

一.问题描述 Android应用中经常涉及从网络中加载大量图片,为提升加载速度和效率,减少网络流量都会采用二级缓存和异步加载机制,所谓二级缓存就是通过先从内存中获取.再从文件中获取,最后才会访问网络.内存缓存(一级)本质上是Map集合以key-value对的方式存储图片的url和Bitmap信息,由于内存缓存会造成堆内存泄露, 管理相对复杂一些,可采用第三方组件,对于有经验的可自己编写组件,而文件缓存比较简单通常自己封装一下即可.下面就通过案例看如何实现网络图片加载的优化. 二.案例介绍 案例新

Android之ListView异步加载图片且仅显示可见子项中的图片

折腾了好多天,遇到 N 多让人崩溃无语的问题,不过今天终于有些收获了,这是实验的第一版,有些混乱,下一步进行改造细分,先把代码记录在这儿吧. 网上查了很多资料,发现都千篇一律,抄来抄去,很多细节和完整实例都没看到,只有自己一点点研究了,总体感觉 android 下面要显示个图片真不容易啊. 项目主要实现的功能: 异步加载图片图片内存缓存.异步磁盘文件缓存解决使用 viewHolder 后出现的图片错位问题优化列表滚动性能,仅显示可见子项中的图片无需固定图片显示高度,对高度进行缓存使列表滚动时不会

cocos2d-x 3.3 之卡牌设计 NO.6 Loading界面(异步加载图片,plist)

刚开始做卡牌的时候没有想到要做loading,因为小游戏资源不多. 但是后来不断的加图片,直到在真机上发现卡顿的问题,我才知道该需要加loading了...... 首先,我们先定义类: class Loading : public Layer { public: bool init(); CREATE_FUNC( Loading); static Scene* CreateScene(); int total_pic_num;//需加载图片数 int total_sound_num;//需加载声

没有网络连接时程序崩溃问题以及动态加载图片问题已解决

经过进一步的研究我们把没有网络连接时程序崩溃的大bug修改掉了,如果是程序打开时没有网络连接会弹出网络连接失败的 对话框,如果在程序运行过程中出现网络异常,在需要连接服务器的时候会抛出网络连接异常: 第二个是动态加载图片问题的解决,我们通过查资料找到方法把bitmap加载进gradeview,用一个线程每次加载一个图片然后 把对应课程位置的图标替换掉,最后实现了动态刷新网络图片. 这两个问题修改过之后,我们的程序中基本的功能已经完善了,最后我们会对UI进行小的修改,让程序更加完美. 下面是新的安

Android-异步加载图片

刚刚复习了之前学的异步加载图片,于是决定写下来,增强记忆 想要加载图片,必须获得所有图片的URL地址.这里图片的URL地址存在了List数组. 异步加载图片有两种方式(我这里使用两种方式):1.自己开线程来加载图片,然后使用Handler来更新UI:2.使用AsyncTask来加载图片. 1.使用子线程来加载图片. 1 private Handler handler = new Handler() 2 { 3 @Override 4 public void handleMessage(Messa

滚屏异步加载图片

<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> <style> </style> </head> <body> <div id="image"> </div> <script src=&qu