现在的项目中不可避免的要使用到网络请求,而且几乎所有软件都有上下拉刷新功能,所以我在此对我的项目进行一个网络请求的封装和上下拉刷新的封装,如果有哪里不对,或更优的地方请在下面留言。
封装的好处:可以让我们的代码更佳精简,代码的耦合程度减少,便于代码的维护更新。
附上展示效果
项目的GitHub地址:https://github.com/cf0717/AFNetWorking-MJRefreash-Encapsulation
下面来介绍我封装代码的思路
使用了AFNetworking和MJRefresh三方工程
Model文件为请求回来后解析的类提供了三个方法(PS:这里我没有使用第三方解析数据,是觉得第三方有局限性,例如:我的字符串与服务器返回的字符串不一致,需要拼接,用三方的话就要在外面拼接,而自己写的话可以直接在解析时拼接,更符合MVC设计模式)
<ul type = "1">
<li>//返回模型 属性方法,不常使用
-(instancetype)initWithDict:(NSDictionary *)dict;
<li>//返回模型 类方法,经常使用
+(instancetype)dataModelWithDict:(NSDictionary *)dict;
<li>//返回数组 类方法,如果数据为数组,那么就可以使用
+(NSMutableArray *)dataModelWithArray:(NSArray *)dataArr;
</ul>
-(instancetype)initWithDict:(NSDictionary *)dict
{
self = [super init];
if (self) {
//解析数据
self.ID = dict[@"id"];
self.titleStr = dict[@"title"];
self.thumbStr = dict[@"thumb"];
self.introStr = dict[@"introduction"];
self.timeStr = dict[@"createAt"];
}
return self;
}
+(instancetype)dataModelWithDict:(NSDictionary *)dict
{
return [[self alloc]initWithDict:dict];
}
+(NSMutableArray *)dataModelWithArray:(NSArray *)dataArr
{
NSMutableArray *dataMutArr = [NSMutableArray array];
for (int i = 0; i < dataArr.count; i++) {
CFDataModel *dataModel = [self dataModelWithDict:dataArr[i]];
[dataMutArr addObject:dataModel];
}
return dataMutArr;
}
Tool文件为主要自定义封装文件
CFSendNet是对网络进行的封装
CFCustomHeader和CFCustomFooter是对上下拉进行的封装
在对网络进行封装的时候,考虑到页面消失时,要把页面的请求给关闭掉这种情况,所以封装起来代码相对少一点,因为请求管理者要是视图控制器的属性。(PS:这种需要在视图管理者中需懒加载网络管理者)
代码如下:
/**
发送网络请求
@param manager 请求管理者
@param urlStr 请求网址
@param parameter 请求参数
@param completion 请求成功返回
@param faild 请求失败返回
*/
+(void)sendNetAFNManger:(AFHTTPSessionManager *)manager urlStr:(NSString *)urlStr parameter:(NSDictionary *)parameter completion:(void(^)(NSDictionary *dict))completion faild:(void(^)(NSError *error))faild
{
[manager GET:urlStr parameters:parameter progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
//请求成功
completion(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
faild(error);
}];
}
/**
结束任务
*
+(void)cancelAFNManger:(AFHTTPSessionManager *)manager
{
[manager.tasks makeObjectsPerformSelector:@selector(cancel)];
}
如果不考虑页面消失时的网络处理,相对而言封装更多一点
代码如下:
/**
发送网络请求
@param urlStr 请求网址
@param parameter 请求参数
@param completion 请求成功返回
@param faild 请求失败返回
*/
+(void)sendNetUrlStr:(NSString *)urlStr parameter:(NSDictionary *)parameter completion:(void(^)(NSDictionary *dict))completion faild:(void(^)(NSError *error))faild
{
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
[manager GET:urlStr parameters:parameter progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
//请求成功
completion(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
faild(error);
}];
}
ViewCntroller文件就是主要的视图了
#pragma mark - 懒加载UITableView和AFHTTPSessionManager
-(UITableView *)tableview
{
if (!_tableview) {
_tableview = [[UITableView alloc]init];
}
return _tableview;
}
-(AFHTTPSessionManager *)manger
{
if (!_manger) {
_manger = [AFHTTPSessionManager manager];
}
return _manger;
}
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor brownColor];
self.tableview.frame = self.view.bounds;
self.tableview.dataSource = self;
self.tableview.rowHeight = 60;
[self.view addSubview:self.tableview];
//TODO:增加上下拉刷新功能
CFCustomHeader *customHeader = [CFCustomHeader customHeaderTarget:self selector:@selector(loadNewData)];
self.tableview.mj_header = customHeader;
CFCustomFooter *customFooter = [CFCustomFooter customFooterTarget:self selector:@selector(loadMoreData)];
self.tableview.mj_footer = customFooter;
[self.tableview.mj_header beginRefreshing];//MJRefresh开始刷新
}
//视图消失就取消请求任务
-(void)viewWillDisappear:(BOOL)animated
{
[CFSendNet cancelAFNManger:self.manger];
}
发送请求时要注意,上拉加载更多失败时页数要减一,否则会遗漏数据。下拉刷新时,数据要清空,不然会有重复数据。请求失败大多为网络不可用,有些时候需要展示一个网络不可用的界面,就是更换界面了。
#pragma mark - 发送请求
-(void)loadNewData
{
[CFSendNet cancelAFNManger:self.manger];
_pageNo = 1;
//1.视图懒加载请求管理者
//2.拼接参数
NSString *kindStr = @"/news";
NSString *urlStr = [NSString stringWithFormat:@"%@%@",httpHead,kindStr];//宏定义头部,拼接字符串
NSMutableDictionary *parameter = [NSMutableDictionary dictionary];
parameter[@"address"] = @"";
parameter[@"rows"] = @"5";
parameter[@"page"] = [NSString stringWithFormat:@"%zd",_pageNo];
//3.发送网路请求
[self.manger GET:urlStr parameters:parameter progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
//TODO:请求成功
if ([responseObject[@"status"] intValue] == 200) {
//数据转模型
_dataArr = [NSMutableArray array];
_dataArr = [CFDataModel dataModelWithArray:responseObject[@"data"]];
[self.tableview reloadData];
}else{
NSLog(@"失败了,错误了");
}
//结束刷新
[self.tableview.mj_header endRefreshing];
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//结束刷新
[self.tableview.mj_header endRefreshing];
//更换界面
NSLog(@"%@",error);
}];
}
//请求更多数据
-(void)loadMoreData
{
[CFSendNet cancelAFNManger:self.manger];
++_pageNo;
//1.视图懒加载请求管理者
//2.拼接参数
NSString *kindStr = @"/news";
NSString *urlStr = [NSString stringWithFormat:@"%@%@",httpHead,kindStr];//宏定义头部,拼接字符串
NSMutableDictionary *parameter = [NSMutableDictionary dictionary];
parameter[@"address"] = @"";
parameter[@"rows"] = @"5";
parameter[@"page"] = [NSString stringWithFormat:@"%zd",_pageNo];
//3.发送网路请求
[self.manger GET:urlStr parameters:parameter progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
//TODO:请求成功
if ([responseObject[@"status"] intValue] == 200) {
//数据转模型
[_dataArr addObjectsFromArray:[CFDataModel dataModelWithArray:responseObject[@"data"]]];
[self.tableview reloadData];
}else{
--_pageNo;
NSLog(@"失败了,错误了");
}
//结束刷新
[self.tableview.mj_footer endRefreshing];
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//结束刷新
[self.tableview.mj_footer endRefreshing];
--_pageNo;
//更换界面
NSLog(@"%@",error);
}];
}
#pragma mark - UITableViewDataSource
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
//这里最好也设置一下
self.tableview.mj_footer.hidden = (_dataArr.count == 0);
return _dataArr.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *cellID = @"cellID";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];
if (!cell) {
cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:cellID];
}
//设置数据
CFDataModel *dataModel = _dataArr[indexPath.row];
cell.textLabel.text = dataModel.titleStr;
cell.detailTextLabel.text = dataModel.introStr;
return cell;
}