iOS项目中的网络请求和上下拉刷新封装
一、运行效果图
现在的项目中不可避免的要使用到网络请求,而且几乎所有软件都有上下拉刷新功能,所以我在此对我的项目进行一个网络请求的封装和上下拉刷新的封装,如果有哪里不对,或更优的地方请在下面留言。
封装的好处:可以让我们的代码更佳精简,代码的耦合程度减少,便于代码的维护更新。
附上展示效果

二、程序实现
下面来介绍我封装代码的思路

使用了AFNetworking和MJRefresh三方工程
Model文件为请求回来后解析的类提供了三个方法(PS:这里我没有使用第三方解析数据,是觉得第三方有局限性,例如:我的字符串与服务器返回的字符串不一致,需要拼接,用三方的话就要在外面拼接,而自己写的话可以直接在解析时拼接,更符合MVC设计模式)
- //返回模型 属性方法,不常使用
-(instancetype)initWithDict:(NSDictionary *)dict; - //返回模型 类方法,经常使用
+(instancetype)dataModelWithDict:(NSDictionary *)dict; - //返回数组 类方法,如果数据为数组,那么就可以使用
+(NSMutableArray *)dataModelWithArray:(NSArray *)dataArr;
```
-(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;
}
> 代码地址如下:<br>http://www.demodashi.com/demo/11621.html
> 注:本文著作权归作者,由demo大师代发,拒绝转载,转载需要作者授权
iOS项目中的网络请求和上下拉刷新封装的更多相关文章
- flutter中的网络请求和下拉刷新上拉加载,toast的案例
添加依赖 pull_to_refresh: ^1.5.6 dio: ^2.1.0 fluttertoast: ^3.0.1 DioUtil import 'package:dio/dio.dart'; ...
- swift中第三方网络请求库Alamofire的安装与使用
swift中第三方网络请求库Alamofire的安装与使用 Alamofire是swift中一个比较流行的网络请求库:https://github.com/Alamofire/Alamofire.下面 ...
- React Native中的网络请求fetch和简单封装
React Native中的网络请求fetch使用方法最为简单,但却可以实现大多数的网络请求,需要了解更多的可以访问: https://segmentfault.com/a/1190000003810 ...
- ios项目中引用其他项目复习
ios项目中引用其他开源项目,今天再次复习了,记个备注. 1. 将开源项目的.xcodeproj拖入项目frameworks 2. Build Phases下 Links Binary With Li ...
- iOS-C文件添加到iOS项目中,运行报错
iOS-C文件添加到iOS项目中,运行报错 问题: 往项目中添加一个空的c文件, 编译运行; 出现2,30个编译错误. 原因: 由于在项目中添加了Pch文件,在文件中所有代码还没有开始运行之前, pc ...
- iOS项目中常见的文件
html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...
- XamarinSQLite教程Xamarin.iOS项目中打开数据库文件
XamarinSQLite教程Xamarin.iOS项目中打开数据库文件 以下是打开MyDocuments.db数据库的具体操作步骤: (1)将Mac电脑上的MyDocuments.db数据库移动到W ...
- XamarinSQLite教程在Xamarin.iOS项目中定位数据库文件
XamarinSQLite教程在Xamarin.iOS项目中定位数据库文件 开发者可以在指定的路径中找到复制的数据库文件,具体的操作步骤如下: (1)单击Mac电脑中Finder菜单中的“前往”|“前 ...
- 在Xamarin.iOS项目中使用预设数据库
在Xamarin.iOS项目中使用预设数据库 当开发者准备好一个预设数据库文件后,就可以将这个数据库文件添加到创建的项目中了.本节将分别在Xamarin.iOS和Xamarin.Android项目中使 ...
随机推荐
- spark热点互动问答
[Spark亚太研究院 决战云计算大数据时代 100期公益大讲堂 互动问答] Q1:我想问,hdfs的namenode挂了,怎么处理? 使用ZooKeeper: 使用Mesos: 使用Yarn: Q2 ...
- Linux 下 UltraEdit 版本 破解 30 天试用限制
原创 http://yhz61010.iteye.com/blog/2319599 rm -rfd ~/.idm/uex rm -rf ~/.idm/*.spl rm -rf /tmp/*.spl 一 ...
- Django CRM查询(一对多,多对多以及相关的反查)
Customer模型: class Customer(models.Model): name = models.CharField(max_length=32) qq = models.CharFie ...
- HDU 1426 Sudoku Killer【DFS 数独】
自从2006年3月10日至11日的首届数独世界锦标赛以后,数独这项游戏越来越受到人们的喜爱和重视. 据说,在2008北京奥运会上,会将数独列为一个单独的项目进行比赛,冠军将有可能获得的一份巨大的奖品— ...
- boost::operators
boost 的 operators 提供了comparison operators.arithmetic operators.operators for iterators 操作.虽然使用 C++ 的 ...
- HDOJ 4961 Boring Sum
Discription Number theory is interesting, while this problem is boring. Here is the problem. Given a ...
- python3 开发面试题(%s和format的区别)5.31
在格式化字符串中有两种方法: 1.%s 2.format 大家常用的是哪一种方法?为什么要用你选的这种方法? 我们先看一个例子: 首先我们定义一个我军需要击杀的恐怖分子的地理坐标为 c=(128,12 ...
- 1.3(Spring MVC学习笔记)数据绑定
一.数据绑定介绍 用户发送过来的数据,只有传递到服务器端的参数上才会起作用. 比如用户输入的用户名和密码要和后台方法中代表用户名和密码的变量关联起来, 从而才能使用用户传递的数据进行一些操作,这样数据 ...
- python 使用mysql示例
安装MySQL驱动 由于MySQL服务器以独立的进程运行,并通过网络对外服务,所以,需要支持Python的MySQL驱动来连接到MySQL服务器.MySQL官方提供了mysql-connector-p ...
- Java中泛型得到T.class
例子: public class Test<T> { public Class<T> getTClass() { return (Class<T>) ((Param ...