我的编程空间,编程开发者的网络收藏夹
学习永远不晚

举例讲解iOS中延迟加载和上拉刷新/下拉加载的实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

举例讲解iOS中延迟加载和上拉刷新/下拉加载的实现

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

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

// 下拉刷新的原理  - (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView  {      if (scrollView.contentOffset.y < - 100) {                    [UIView animateWithDuration:1.0 animations:^{                            //  frame发生偏移,距离顶部150的距离(可自行设定)              self.tableView.contentInset = UIEdgeInsetsMake(150.0f, 0.0f, 0.0f, 0.0f);          } completion:^(BOOL finished) {                                        }];      }  }    // 上拉加载的原理  - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate  {            NSLog(@"%f",scrollView.contentOffset.y);      NSLog(@"%f",scrollView.frame.size.height);      NSLog(@"%f",scrollView.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) {                                      }];        }  } 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

举例讲解iOS中延迟加载和上拉刷新/下拉加载的实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

举例讲解iOS中延迟加载和上拉刷新/下拉加载的实现

lazy懒加载(延迟加载)UITableView 举个例子,当我们在用网易新闻App时,看着那么多的新闻,并不是所有的都是我们感兴趣的,有的时候我们只是很快的滑过,想要快速的略过不喜欢的内容,但是只要滑动经过了,图片就开始加载了,这样用户体
2022-06-02

Android RecyclerView实现下拉刷新和上拉加载

RecyclerView已经出来很久了,许许多多的项目都开始从ListView转战RecyclerView,那么,上拉加载和下拉刷新是一件很有必要的事情。 在ListView上,我们可以通过自己添加addHeadView和addFootVi
2022-06-06

android使用PullToRefresh实现下拉刷新和上拉加载

PullToRefresh是一套实现非常好的下拉刷新库,它支持: 1.ListView 2.ExpandableListView 3.GridView 4.WebView 等多种常用的需要刷新的View类型,而且使用起来也十分方便。 dem
2022-06-06

Android中使用RecyclerView实现下拉刷新和上拉加载

推荐阅读:使用RecyclerView添加Header和Footer的方法 RecyclerView的使用之HelloWorld RecyclerView 是Android L版本中新添加的一个用
2022-06-06

Android RecyclerView实现下拉刷新和上拉加载更多

使用官方的刷新控件SwipeRefreshLayout来实现下拉刷新,当RecyclerView滑到底部实现下拉加载(进度条效果用RecyclerView加载一个布局实现) 需要完成控件的下拉监听和上拉监听,其中,下拉监听通过SwipRef
2022-06-06

Android控件PullRefreshViewGroup实现下拉刷新和上拉加载

本文实例为大家分享了Android实现下拉刷新和上拉加载更多的具体代码,供大家参考,具体内容如下 先分享下源码:Android实现下拉刷新和上拉加载更多 实现思路:由PullRefreshViewGroup控件来接管标准控件(比如Recyc
2022-06-06

Spring怎么实现上拉刷新和下拉加载效果

这篇文章主要介绍Spring怎么实现上拉刷新和下拉加载效果,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!导依赖:compile com.android.support:recyclerview-v7:25.3.1 c
2023-05-30

uniapp怎么实现下拉刷新和上拉加载功能

随着移动端开发的不断升级,开发者们对于移动应用的需求也越来越高。而在很多移动应用中,下拉刷新和上拉加载更多是必不可少的功能之一,为了提高用户体验,许多移动应用都会加入这两个功能。在这里,我们将介绍如何在uniapp中实现下拉刷新和上拉加载更多的功能。一、基础内容介绍uniapp是基于Vue.js框架的一款跨平台应用开发框架,支持一次编写,多端发布。由于其良好的跨平台特性,同时内
2023-05-14

Android ListView实现上拉加载更多和下拉刷新功能

本文实例为大家介绍了Android ListView下拉刷新功能的实现方法和功能,供大家参考,具体内容如下 1、ListView优化方式 界面缓存:ViewHolder+convertView 分页加载:上拉刷新 图片缓存 快速滑动List
2022-06-06

原生js怎么实现下拉刷新和上拉加载更多

本篇文章为大家展示了原生js怎么实现下拉刷新和上拉加载更多,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。JavaScript是什么JS是JavaScript的简称,它是一种直译式的脚本语言,其解释器
2023-06-26

Android中Listview下拉刷新和上拉加载更多的多种实现方案

listview经常结合下来刷新和上拉加载更多使用,本文总结了三种常用到的方案分别作出说明。 方案一:添加头布局和脚布局 android系统为listview提供了addfootview和addheadview两
2022-06-06

Android ListView实现上拉加载下拉刷新和滑动删除功能

最近项目需要用到可以滑动删除并且带有上拉加载下拉刷新的Listview,查阅了一些资料,大多都是在SwipeMenuListView的基础上去添加头部和底部View,来扩展上拉加载和下拉刷新的功能,不过需要手动的去绘制UI及处理一些动画效果
2022-06-06

uniapp实现下拉刷新与上拉触底加载功能的示例代码

这篇文章主要记录一下uniapp实现下拉刷新与上拉触底加载功能的示例代码,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
2023-05-16

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录