[BS-23] AFN网络请求上拉/下拉刷新的细节问题总结
上拉/下拉刷新的细节问题总结
1.如果导航栏有透明色,则也需要设置header自动改变透明度
self.tableView.mj_header.automaticallyChangeAlpha = YES; //允许自动改变透明度
2. 下拉刷新必须手动调用[self.tableView.mj_header beginRefreshing];才开始刷新,下拉刷新只要用户滚动到最下方,自动开始加载更多。
3. 上拉刷新通常用的是用MJRefreshAutoNormalFooter,该控件总是紧贴最后一条数据下方。故第一次进入界面时,还没有数据,footer就会显示在最顶部,比较难看。解决的办法:因为第一次进入tableView,不管有没有从网络加载数据,它都会先调用数据源方法numberOfRowsInSection,且以后每次reloadData也会调用该方法。所以在该方法中控制footer是否隐藏最合适。
#pragma mark - Table view data source
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    //第一次进来或者每次reloadData否会调一次该方法,在此控制footer是否隐藏
    self.tableView.mj_footer.hidden = (self.topics.count == );
    return self.topics.count;
}
4.
在上拉刷新中处理数据采用如下方法: //字典-》模型
self.topics = [WZTopic mj_objectArrayWithKeyValuesArray:responseObject[@"list"]];
分析机理:
如果第二次下拉,返回的数据会将整个self.topics数据覆盖掉,相当于先清空容器中数据,然后再向容器中加入新数据。如果采用self.topics=nil;这样的方式清空,堆中的原来的模型数组(容器)因无强引用就会销毁。然后再调用 [self.topics addObject: dict];因self.topics此前为nil,就会懒加载重新开辟一块内存用来充当容器。这样将原来的罐子摔烂,买个新罐子装东西的做法实属浪费,这样加大系统的负荷,效率没有前者高。
5.通过thisPage = self.page+1来加载服务器数据的细节控制(红色为核心控制代码)
#import "WZWordTableViewController.h"
#import <AFNetworking.h>
#import <UIImageView+WebCache.h> //分类
#import "WZTopic.h"
#import <MJExtension.h>
#import <SVProgressHUD.h>
#import <MJRefresh.h> @interface WZWordTableViewController ()
//服务器返回的值
@property (strong,nonatomic) NSNumber *count; //服务器返回的总数据条数
@property (strong,nonatomic) NSMutableArray *topics; //模型数组 //请求参数
@property (assign,nonatomic) NSInteger page;//发送请求时使用的页数,每次加载下一页,让该值page +1 //保存本次发送请求的参数,用于控制只处理最后一次请求的参数(如下拉不成功,又去上拉,上拉的数据显示出来后,下拉的数据才回来,此时会将表中已显示数据突然清空,只显示第一页数据)
@property (strong,nonatomic) NSDictionary *newestParams; @end @implementation WZWordTableViewController - (void)viewDidLoad {
[super viewDidLoad]; self.tableView.rowHeight = ; //设置下拉和上拉刷新
[self setupMJRefresh]; } //懒加载topics的get方法,第一次调self.topics时,分配空间和初始化topics数组
- (NSMutableArray *)topics { if (!_topics) {
_topics = [[NSMutableArray alloc]init];
} return _topics;
} /** 设置下拉和上拉刷新 */
- (void)setupMJRefresh { //设置header
self.tableView.mj_header = [MJRefreshNormalHeader headerWithRefreshingTarget:self refreshingAction:@selector(loadNewTopics)];
//允许自动改变透明度
self.tableView.mj_header.automaticallyChangeAlpha = YES;
//header开始刷新
[self.tableView.mj_header beginRefreshing]; //设置footer,用户滑到最下边就会自动启用footer刷新,故不用写开始刷新的代码
self.tableView.mj_footer = [MJRefreshAutoNormalFooter footerWithRefreshingTarget:self refreshingAction:@selector(loadMoreTopics)]; //self.tableView.mj_footer.hidden = YES;//数据尚未返回,或者加载失败footer都会跑到最顶部。//改成在numberOfRows方法中控制了
} /** 加载新的帖子 */
- (void)loadNewTopics { //下拉先停止footer刷新
[self.tableView.mj_footer endRefreshing]; //每次下拉让页码都回归到第0页
//self.page = 0; (不在此设置,如多次下拉加载不成功,在回到底部上拉,此时页码已变为0,之前可能已是第5页了,后面又接着1,2,3...出现重复加载) //设置参数
NSMutableDictionary *params = [NSMutableDictionary dictionary];
params[@"a"] = @"list";
NSInteger thisPage = 0; //第一次请求发送0还是1由看接口文档,此处不能直接写self.page=0; 每次请求成功后,才能将当前thisPage保存进self.page中,供下次请求使用。
params[@"page"] = @(thisPage); //此处不能写为@(self.page), self.newestParams = params; //self.newestParams是全局的,永远只存储最后一次请求的参数,之前的请求参数都会被最后一次覆盖掉。 //发送请求
[[AFHTTPSessionManager manager] GET:@"http://api.budejie.com/api/api_open.php" parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { //如果不是用户最近一次发送的请求,就不处理,直接return整个AFN方法,后面failure也不会执行
if(params != self.newestParams) return; //能来到这说明本次加载成功,此时将本次页码保存在self.page中,以便下次加载使用。
self.page = thisPage; //解析服务器返回数据
NSDictionary *infoDict = responseObject[@"info"];
self.count = infoDict[@"count"]; //总帖子数可能随时会增加,每次请求都重新存储一遍 //字典-》模型(如果第二次下拉,返回的数据会将整个self.topics数据覆盖掉,相当于先清空,再加载)
self.topics = [WZTopic mj_objectArrayWithKeyValuesArray:responseObject[@"list"]];//此句会在内部通过KVC将responseObject[@"list"]这个包含20个字典的数组转为包含20个topic模型的数组 [self.tableView reloadData];//刷新表格才会显示数据 //返回数据了就停止刷新
[self.tableView.mj_header endRefreshing]; //让footer显示出来(第一次让footer隐藏了)
//self.tableView.mj_footer.hidden = NO; //改成在numberOfRows方法中控制了 } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { //提示用户加载失败
[SVProgressHUD setDefaultMaskType:SVProgressHUDMaskTypeBlack];//第一次使用时设置就可以
[SVProgressHUD showErrorWithStatus:@"加载失败"]; //停止header刷新(第一次加载失败,footer就不会显示出来,不用停止footer)
[self.tableView.mj_header endRefreshing]; }]; } /** 加载更多帖子 */
- (void)loadMoreTopics { //上拉先停止header刷新
[self.tableView.mj_header endRefreshing]; //设置参数
NSMutableDictionary *params = [NSMutableDictionary dictionary];
params[@"a"] = @"list";
NSInteger thisPage = (self.page+1); //此处不能写成++self.page,会改变self.page的值。如果加载不成功,还得去--。每次请求成功后,才能将当前thisPage保存进self.page中,供下次请求使用。
params[@"page"] = @(thisPage);
self.newestParams = params; //self.newestParams是全局的,永远只存储最后一次请求的参数,之前的请求参数都会被最后一次覆盖掉。 //发送请求
[[AFHTTPSessionManager manager] GET:@"http://api.budejie.com/api/api_open.php" parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { //如果不是用户最近一次发送的请求,就不处理,直接return整个AFN方法,后面failure也不会执行
if(params != self.newestParams) return; //能来到这说明本次加载成功,此时将本次页码保存在self.page中,以便下次加载使用。
self.page = thisPage; //解析服务器返回数据
NSDictionary *infoDict = responseObject[@"info"];
self.count = infoDict[@"count"]; //总帖子数可能随时会增加,每次请求都重新存储一遍 //字典-》模型
NSArray *array = [WZTopic mj_objectArrayWithKeyValuesArray:responseObject[@"list"]];//此句会在内部通过KVC将responseObject[@"list"]这个包含20个字典的数组转为包含20个topic模型的数组 //添加模型到可变数字topics中
[self.topics addObjectsFromArray:array];//把第二次返回的数据加到数组中 [self.tableView reloadData];//刷新表格才会显示数据 } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { //提示用户加载失败
//[SVProgressHUD setDefaultMaskType:SVProgressHUDMaskTypeBlack];//只需要设置一次就ok
[SVProgressHUD showErrorWithStatus:@"加载失败"]; //加载失败了footer停止刷新
[self.tableView.mj_footer endRefreshing]; }];
} #pragma mark - Table view data source - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { //判断footer是否隐藏,判断footer停止刷新时显示"加载更多"还是"全部数据已加载完成"
[self checkFooterStatus]; return self.topics.count;
} //判断footer是否隐藏,判断footer停止刷新时显示"加载更多"还是"全部数据已加载完成"
-(void)checkFooterStatus {
//第一次进来或者每次reloadData否会调一次该方法,在此控制footer是否隐藏
self.tableView.mj_footer.hidden = (self.topics.count == ); //用户滑到最下边就会自动启用footer刷新,现在数据回来了要footer停止刷新
if (self.topics.count == self.count.unsignedIntegerValue) {
[self.tableView.mj_footer endRefreshingWithNoMoreData];
} else {
[self.tableView.mj_footer endRefreshing];
}
} - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
//创建cell
static NSString *const ID = @"Word";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
if (cell == nil) {
cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID];
} //获取模型
WZTopic *topic = self.topics[indexPath.row]; //设置数据
cell.textLabel.text = topic.name;
cell.detailTextLabel.text = topic.text;
[cell.imageView sd_setImageWithURL:[NSURL URLWithString:topic.profile_image] placeholderImage:[UIImage imageNamed:@"defaultUserIcon"]]; //返回cell
return cell;
} @end
6.通过最大的id来加载服务器数据的细节控制(删掉恶心的page控制代码,直接发给服务器最大id即可加载下一页,红色为核心控制代码)
#import "WZWordTableViewController.h"
#import <AFNetworking.h>
#import <UIImageView+WebCache.h> //分类
#import "WZTopic.h"
#import <MJExtension.h>
#import <SVProgressHUD.h>
#import <MJRefresh.h> @interface WZWordTableViewController ()
//服务器返回的值
@property (strong,nonatomic) NSNumber *count; //服务器返回的总数据条数
@property (strong,nonatomic) NSMutableArray *topics; //模型数组 //保存本次发送请求的参数,用于控制只处理最后一次请求的参数(如下拉不成功,又去上拉,上拉的数据显示出来后,下拉的数据才回来,此时会将表中已显示数据突然清空,只显示第一页数据)
@property (strong,nonatomic) NSDictionary *newestParams; @end @implementation WZWordTableViewController - (void)viewDidLoad {
[super viewDidLoad]; self.tableView.rowHeight = ; //设置下拉和上拉刷新
[self setupMJRefresh]; } //懒加载topics的get方法,第一次调self.topics时,分配空间和初始化topics数组
- (NSMutableArray *)topics { if (!_topics) {
_topics = [[NSMutableArray alloc]init];
} return _topics;
} /** 设置下拉和上拉刷新 */
- (void)setupMJRefresh { //设置header
self.tableView.mj_header = [MJRefreshNormalHeader headerWithRefreshingTarget:self refreshingAction:@selector(loadNewTopics)];
//允许自动改变透明度
self.tableView.mj_header.automaticallyChangeAlpha = YES;
//header开始刷新
[self.tableView.mj_header beginRefreshing]; //设置footer,用户滑到最下边就会自动启用footer刷新,故不用写开始刷新的代码
self.tableView.mj_footer = [MJRefreshAutoNormalFooter footerWithRefreshingTarget:self refreshingAction:@selector(loadMoreTopics)]; //self.tableView.mj_footer.hidden = YES;//数据尚未返回,或者加载失败footer都会跑到最顶部。//改成在numberOfRows方法中控制了
} /** 加载新的帖子 */
- (void)loadNewTopics { //下拉先停止footer刷新
[self.tableView.mj_footer endRefreshing]; //设置参数
NSMutableDictionary *params = [NSMutableDictionary dictionary];
params[@"a"] = @"list";
//params[@"maxtime"] = @""; //不设置maxtime默认加载第一页
self.newestParams = params; //self.newestParams是全局的,永远只存储最后一次请求的参数,之前的请求参数都会被最后一次覆盖掉。 //发送请求
[[AFHTTPSessionManager manager] GET:@"http://api.com/api/api_open.php" parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { //如果不是用户最近一次发送的请求,就不处理,直接return整个AFN方法,后面failure也不会执行
if(params != self.newestParams) return; //解析服务器返回数据
NSDictionary *infoDict = responseObject[@"info"];
self.count = infoDict[@"count"]; //总帖子数可能随时会增加,每次请求都重新存储一遍
self.maxtime = infoDict[@"maxtime"]; //下次加载需使用 //字典-》模型(如果第二次下拉,返回的数据会将整个self.topics数据覆盖掉,相当于先清空,再加载)
self.topics = [WZTopic mj_objectArrayWithKeyValuesArray:responseObject[@"list"]];//此句会在内部通过KVC将responseObject[@"list"]这个包含20个字典的数组转为包含20个topic模型的数组 [self.tableView reloadData];//刷新表格才会显示数据 //返回数据了就停止刷新
[self.tableView.mj_header endRefreshing]; //让footer显示出来(第一次让footer隐藏了)
//self.tableView.mj_footer.hidden = NO; //改成在numberOfRows方法中控制了 } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { //提示用户加载失败
[SVProgressHUD setDefaultMaskType:SVProgressHUDMaskTypeBlack];//第一次使用时设置就可以
[SVProgressHUD showErrorWithStatus:@"加载失败"]; //停止header刷新(第一次加载失败,footer就不会显示出来,不用停止footer)
[self.tableView.mj_header endRefreshing]; }]; } /** 加载更多帖子 */
- (void)loadMoreTopics { //上拉先停止header刷新
[self.tableView.mj_header endRefreshing]; //设置参数
NSMutableDictionary *params = [NSMutableDictionary dictionary];
params[@"a"] = @"list";
params[@"maxtime"] = self.maxtime; //由上一次返回的数据提供
self.newestParams = params; //self.newestParams是全局的,永远只存储最后一次请求的参数,之前的请求参数都会被最后一次覆盖掉。 //发送请求
[[AFHTTPSessionManager manager] GET:@"http://api.com/api/api_open.php" parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { //如果不是用户最近一次发送的请求,就不处理,直接return整个AFN方法,后面failure也不会执行
if(params != self.newestParams) return; //解析服务器返回数据
NSDictionary *infoDict = responseObject[@"info"];
self.count = infoDict[@"count"]; //总帖子数可能随时会增加,每次请求都重新存储一遍
self.maxtime = infoDict[@"maxtime"]; //下次加载需使用 //字典-》模型
NSArray *array = [WZTopic mj_objectArrayWithKeyValuesArray:responseObject[@"list"]];//此句会在内部通过KVC将responseObject[@"list"]这个包含20个字典的数组转为包含20个topic模型的数组 //添加模型到可变数字topics中
[self.topics addObjectsFromArray:array];//把第二次返回的数据加到数组中 [self.tableView reloadData];//刷新表格才会显示数据 } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { //提示用户加载失败
//[SVProgressHUD setDefaultMaskType:SVProgressHUDMaskTypeBlack];//只需要设置一次就ok
[SVProgressHUD showErrorWithStatus:@"加载失败"]; //加载失败了footer停止刷新
[self.tableView.mj_footer endRefreshing]; }];
} #pragma mark - Table view data source - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { //判断footer是否隐藏,判断footer停止刷新时显示"加载更多"还是"全部数据已加载完成"
[self checkFooterStatus]; return self.topics.count;
} //判断footer是否隐藏,判断footer停止刷新时显示"加载更多"还是"全部数据已加载完成"
-(void)checkFooterStatus {
//第一次进来或者每次reloadData否会调一次该方法,在此控制footer是否隐藏
self.tableView.mj_footer.hidden = (self.topics.count == ); //用户滑到最下边就会自动启用footer刷新,现在数据回来了要footer停止刷新
if (self.topics.count == self.count.unsignedIntegerValue) {
[self.tableView.mj_footer endRefreshingWithNoMoreData];
} else {
[self.tableView.mj_footer endRefreshing];
}
} - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
//创建cell
static NSString *const ID = @"Word";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
if (cell == nil) {
cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID];
} //获取模型
WZTopic *topic = self.topics[indexPath.row]; //设置数据
cell.textLabel.text = topic.name;
cell.detailTextLabel.text = topic.text;
[cell.imageView sd_setImageWithURL:[NSURL URLWithString:topic.profile_image] placeholderImage:[UIImage imageNamed:@"defaultUserIcon"]]; //返回cell
return cell;
} @end
文章系作者原创,转载请注明出处:http://www.cnblogs.com/stevenwuzheng/p/5506286.html
如有错误,欢迎随时指正!
[BS-23] AFN网络请求上拉/下拉刷新的细节问题总结的更多相关文章
- Swiper 判断上滑下拉操作
		
onTouchMove: function(swiper){ //手动滑动中触发//判断上滑下拉var i = mySwiper.translate;setTimeout(function() {va ...
 - ios position:fixed 上滑下拉抖动
		
ios position:fixed 上滑下拉抖动 最近呢遇到一个ios的兼容问题,界面是需要一个头底部的固定的效果,用的position:fixed定位布局,写完测试发现安卓手机正常的,按时ios上 ...
 - ASI与AFN网络请求的的比较
		
对比 ASI AFN 更新状态 2012年10月份,已经停止更新 持续更新中,目前已更新至3.0版 介绍 ASI的直接操作对象ASIHTTPRequest,是一个实现了了NSCopying协议的NSO ...
 - 原生网络请求以及AFN网络请求/异步下载
		
这里对网络请求方式做一个总结. 原生方式同步GET请求: NSString *urlStr = @"http://apis.juhe.cn/mobile/get?phone=13429667 ...
 - jsp页面上的下拉框案例(Struts2)
		
<s:select></s:select>包含的属性有:list="" :name="" :value="" ...
 - 浅谈对MJRefresh(上)下拉刷新控件的理解
		
MJRefresh GitHub地址:https://github.com/CoderMJLee/MJRefresh 利用业余时间研究了一下iOS的开发,发现OC特定的语法方式吸引了我,而且iOS开发 ...
 - 160823、ionic上拉/下拉更新数据
		
<!DOCTYPE html> <html ng-app="myApp"> <head> <meta charset="UTF- ...
 - js实现页面的上滑下拉功能
		
这两天做项目,用到了上滑和下拉的功能,主要是通过监听touchmove,touchstart,touchend三个事件去判断页面上滑状态还是下拉状态. 同时加一个知识点:有时在监听时会报错,这个错是这 ...
 - SDI在自定义的工具栏上添加下拉控件
		
0.首先到自己的工具条上新建一个控件,并命名新ID 1.拷贝FlatComboBox.h和FlatComboBox.cpp到工程目录下 2.建立新类 class CTrackerToolBar : p ...
 
随机推荐
- [排错] Status error 2850
			
前几天重新搭建了APAC的Netbackup, 终于可以备份成功了, 但是今天在做还原的时候遇到了些小问题,记录下来. 1. 在CNHZSRV04BPO上执行还原的时候报下面的错误. 2. 经过检查发 ...
 - java字符串练习题
 - Andrew Ng机器学习公开课笔记 – Factor Analysis
			
网易公开课,第13,14课 notes,9 本质上因子分析是一种降维算法 参考,http://www.douban.com/note/225942377/,浅谈主成分分析和因子分析 把大量的原始变量, ...
 - Delphi 如何清除动态数组的内存?
			
SetLength(glb_IndexConfig,); FreeAndNil(glb_IndexConfig);
 - SQLite(快速上手版)笔记
			
原文 1. SQL语法关键字 关键字 描述 Create Table 创建数据表 Alter Table 修改数据表 Drop Table 删除数据表 Create Index 创建索引 Drop I ...
 - url匹配和match()方法
			
下面的全局匹配可以找到字符串中的所有数字: "1 plus 2 equals 3".match(/\d+/g) // 返回 ["1", "2" ...
 - Bluetooth SDP介绍
			
目录 1. 概念 2. 服务记录(Service Record) 3. 服务属性(Service Attribute) 4. 服务类(Service Class) 5. 服务查找 5.1 UUID 5 ...
 - JSON数据转换为字典型
			
-(NSDictionary *)parseJSONStringToNSDictionary:(NSString *)JSONString { NSData *JSONData = [JSONS ...
 - wget ftp
			
今天操作远端机器的时候发现少一个安装包, 需要传到对方的机器上,还能使用通过的老办法,直接SSH连上去了,发现传的很慢, 只有40K的样子, 看时间还需要二个多小时就有点受不了了.想想有一台FTP服务 ...
 - wpf前端设计
			
http://www.cnblogs.com/w-wanglei/archive/2016/03/14/5274298.html#_nav_0