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

@图片下载类  

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

  

}

下拉刷新和上拉加载的原理
很多App中,新闻或者展示类都存在下拉刷新和上拉加载的效果,网上提供了实现这种效果的第三方类(详情请见MJRefresh和EGOTableViewPullRefresh),用起来很方便,但是闲暇之余,我们可以思考下,这种效果实现的原理是什么,我以前说过,只要是动画都是骗人的,只要不是硬件问题大部分效果都能在系统UI的基础上做出来.
下面是关键代码分析:

// 下拉刷新的原理  

- (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView  

{  

    if (scrollView.contentOffset.y  

     *  scrollView一开始并不存在偏移量,但是会设定contentSize的大小,所以contentSize.height永远都会比contentOffset.y高一个手机屏幕的 

     *  高度;上拉加载的效果就是每次滑动到底部时,再往上拉的时候请求更多,那个时候产生的偏移量,就能让contentOffset.y   手机屏幕尺寸高大于这 

     *  个滚动视图的contentSize.height 

     */  

    if (scrollView.contentOffset.y   scrollView.frame.size.height >= scrollView.contentSize.height) {  

          

        NSLog(@"%d %s",__LINE__,__FUNCTION__);  

        [UIView commitAnimations];  

          

        [UIView animateWithDuration:1.0 animations:^{  

            //  frame发生的偏移量,距离底部往上提高60(可自行设定)  

            self.tableView.contentInset = UIEdgeInsetsMake(0, 0, 60, 0);  

        } completion:^(BOOL finished) {  

              

            /** 

             *  发起网络请求,请求加载更多数据 

             *  然后在数据请求回来的时候,将contentInset改为(0,0,0,0) 

             */  

        }];  

  

    }  

} 

举例讲解iOS中延迟加载和上拉刷新/下拉加载的实现的更多相关文章

  1. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  2. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  4. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  5. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  6. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

  7. ios – 什么控制iTunes中iPhone应用程序支持的语言列表?

    什么控制iPhone应用程序的iTunes页面中支持的语言?

  8. ios – 获得APNs响应BadDeviceToken或Unregistered的可能原因是什么?

    我知道设备令牌在某些时候是有效的.用户如何使其设备令牌变坏?从关于“未注册”的文档:Thedevicetokenisinactiveforthespecifiedtopic.这是否意味着应用程序已被删除?.您应该看到四种分发方法:如果您选择AppStore或Enterprise,您将在后面的对话框中看到Xcode将APNS权利更改为生产:如果选择AdHoc或Development,则aps-environment下的文本将是开发,然后应与后端的配置匹配.

  9. ios – 当我关闭应用程序时,我从调试器获得消息:由于信号15而终止

    我怎么能解决这个问题,我不知道这个链接MypreviousproblemaboutCoredata对我的问题有影响吗?当我cmd应用程序的Q时,将出现此消息.Messagefromdebugger:Terminatedduetosignal15如果谁知道我以前的问题的解决方案,请告诉我.解决方法>来自调试器的消息:每当用户通过CMD-Q(退出)或STOP手动终止应用程序(无论是在iOS模拟器中还是

  10. ios – NSUbiquityIdentityDidChangeNotification和SIGKILL

    当应用程序被发送到后台时,我们会删除观察者吗?我遇到的问题是,当UbiquityToken发生变化时,应用程序终止,因为用户已经更改了iCloud设置.你们如何设法订阅这个通知,如果你不这样做,你会做什么来跟踪当前登录的iCloud用户?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部