[10分钟教你]封装iOS网络服务
联系方式:401518215@qq.com
童稚们,我有回来了> <。感觉上次写的教程反响还是挺好的,我动力十足呀。今天我也来带给大家在iOS项目开发里面必须的技能--网络请求。当然啦,普通的网络请求写起来不困难,本来就属于iOS开发的必修课程,不过现在我们需要考虑的是处于项目开发的环境下,自然就少不了对网络服务的封装,不然我们在项目的使用过程中就变得异常的混乱和长~~~~。
不多说,我们先要导入一个轮子:AFNetworking。不得不说这个第三方库的功能实在强大,今天我们使用到的网络请求功能也不过是这个库的凤毛麟角,想要仔细学习的话可以自己去看一下。导入方式不多说:拷贝或者cocoapod。
必须说明的是AFNetworking其实已经把网络请求给封装一次了,那为什么我们还要封装一次呢?只是为了统一我们的请求接口和使用于我们项目的实质需求。毕竟别人封装考虑的是全面和实用,而我们在他们的基础上再次封装则是为了我们的项目开发需求。
说了这么多,终于要上代码了。先创建两个网络服务类
typedef NS_ENUM(NSUInteger, HttpRequestType) {//get请求
HttpRequestType_Get = 0,
//post请求
HttpRequestType_Post
};
typedef NS_ENUM(NSInteger, AFNetworkErrorType) {
AFNetworkErrorType_TimeOut = NSURLErrorTimedOut,//连接超时
AFNetworkErrorType_UnURL = NSURLErrorUnsupportedURL,//无URL
AFNetworkErrorType_NoNetwork = NSURLErrorNotConnectedToInternet,//无网络
AFNetworkErrorType_404Failed = NSURLErrorBadServerResponse,//404
AFNetworkErrorType_3840Failed = 3840//网络错误
};
@interface CHHttpRequest : NSObject/**
*发送get请求
*
*@paramURLString请求的网址字符串
*@paramparameters 请求的参数
*@paramsuccess请求成功的回调
*@paramfailure请求失败的回调
*/+(void)getWithURLString:(NSString*)URLString
parameters:(id)parameters
success:(void(^)(id responerObject))success
failure:(void(^)(NSError* error))failure;
/**
*发送post请求
*
*@paramURLString请求的网址字符串
*@paramparameters 请求的参数
*@paramsuccess请求成功的回调
*@paramfailure请求失败的回调
*/
+(void)postWithURLString:(NSString*)URLString
parameters:(id)parameters
success:(void(^)(id responerObject))success
failure:(void(^)(NSError *error))failure;
/**
*发送网络请求
*
*@paramURLString请求的网址字符串
*@paramparameters 请求的参数
*@paramtype请求的类型
*@paramsuccess请求成功的回调
*@paramfailure请求失败的回调
*/+(void)requestWithURLString:(NSString*)URLString
parameters:(id)parameters
type:(HttpRequestType)type
success:(void(^)(id responseObject))success
failure:(void(^)(NSError* error))failure;
/**
*上传图片
*
*@paramURLString请求的网址字符串
*@paramparameters 请求的参数
*@paramuploadParam上传图片的信息
*@paramsuccess请求成功的回调
*@paramfailure请求失败的回调
*/+(void)uploadWithURLString:(NSString *)URLString
parameters:(id)parameters
uploadParam:(CHUploadParam*)uploadParam
success:(void(^)())success
failure:(void(^)(NSError* error))failure;
@end
#import "CHHttpRequest.h"
#import "AFNetworking.h"
#import "CHUploadParam.h"@implementation CHHttpRequest#pragma mark -- GET请求
+(void)getWithURLString:(NSString *)URLString parameters:(id)parameters success:(void (^)(id))success failure:(void (^)(NSError *))failure
{
AFHTTPSessionManager* manager = [AFHTTPSessionManager manager];
//可以接受的类型
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
//请求队列的最大并发数
//manager.operationQueue.maxConcurrentOperationCount = 5;
//请求超时的时间
//manager.requestSerializer.timeoutInterval = 5;
[manager GET:URLString parameters:nil progress:nil success:^(NSURLSessionDataTask* _Nonnull task,id _Nullable responseObject) {
//请求成功
success(responseObject);
}failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
if (failure) {
failure(error);
//输出网络诊断信息
__weak id weakSelf = self;
[weakSelf requestFailure:error];
}
}];
}#pragma mark -- POST请求
+(void)postWithURLString:(NSString *)URLString parameters:(id)parameters success:(void (^)(id))success failure:(void (^)(NSError *))failure
{
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
//请求队列的最大并发数
//manager.operationQueue.maxConcurrentOperationCount = 5;
//请求超时的时间
//manager.requestSerializer.timeoutInterval = 5;
[manager POST:URLString parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id_Nullable responseObject) {
//请求成功
success(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
if (failure) {
failure(error);
//输出网络诊断信息
__weak id weakSelf = self;
[weakSelf requestFailure:error];
}
}];
}#pragma mark -- REQUEST请求
+(void)requestWithURLString:(NSString *)URLString parameters:(id)parameters type:(HttpRequestType)type success:(void (^)(id))success failure:(void (^)(NSError *))failure
{
AFHTTPSessionManager* manager = [AFHTTPSessionManager manager];
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
//请求队列的最大并发数
//manager.operationQueue.maxConcurrentOperationCount = 5;
//请求超时的时间
//manager.requestSerializer.timeoutInterval = 5;
switch (type) {
case HttpRequestType_Get:
{
[manager GET:URLString parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id_Nullable responseObject) {
//请求成功
success(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
if (failure) {
failure(error);
//输出网络诊断信息
__weak id weakSelf = self;
[weakSelf requestFailure:error];
}
}];
}
break;
case HttpRequestType_Post:
{
[manager POST:URLString parameters:parameters progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id_Nullable responseObject) {
//请求成功
success(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
if (failure) {
failure(error);
//输出网络诊断信息
__weak id weakSelf = self;
[weakSelf requestFailure:error];
}
}];
}
break;
}
}#pragma mark -- 上传图片
+(void)uploadWithURLString:(NSString *)URLString parameters:(id)parameters uploadParam:(CHUploadParam *)uploadParam success:(void (^)())success failure:(void (^)(NSError *))failure
{
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
//请求队列的最大并发数
//manager.operationQueue.maxConcurrentOperationCount = 5;
//请求超时的时间
//manager.requestSerializer.timeoutInterval = 5;
[manager POST:URLString parameters:parameters constructingBodyWithBlock:^(id_Nonnull formData) {
[formData appendPartWithFileData:uploadParam.data name:uploadParam.name fileName:uploadParam.fileName mimeType:uploadParam.mimeType];
} progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id_Nullable responseObject) {
//请求成功
success(responseObject);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//请求失败
if (failure) {
failure(error);
//输出网络诊断信息
__weak id weakSelf = self;
[weakSelf requestFailure:error];
}
}];
}#pragma mark -- requestFailure
-(void)requestFailure:(NSError *)error
{
switch (error.code) {
case AFNetworkErrorType_TimeOut:
NSLog(@"网络连接超时");
break;
case AFNetworkErrorType_UnURL:
NSLog(@"没有找到该网络地址");
break;
case AFNetworkErrorType_NoNetwork:
NSLog(@"没有网络服务");
break;
case AFNetworkErrorType_404Failed:
NSLog(@"404网络原因");
break;
case AFNetworkErrorType_3840Failed:
NSLog(@"服务器出错");
break;
default:
NSLog(@"不明原因");
break;
}
}@end
有个CHUploadParam的类是用来处理图片上传的,大家也要创建这个类的.h/.m,代码如下:
@interface CHUploadParam : NSObject
//图片二进制数据
@property (nonatomic,strong) NSData * data;
//服务器对应的参数名称
@property (nonatomic,copy) NSString * name;
//文件的名称
@property (nonatomic,copy) NSString * fileName;
//文件的MIME类型
@property (nonatomic,copy) NSString * mimeType;
@end
在这里以上的代码是重要的,基本上对我们要使用的AFNetworking里面的内容的封装了。大家需要好好理解一下代码,我也都一一注释了。
那么这样的封装就能满足我们的实际使用需求吗?明显是不行的。虽然我们的使用方便了,但是在实际使用的时候,比如一个Controller,我们需要请求网络图片数据,那么我们就需要在里面写我们的网络请求,写json解析,写UI使用。omg,多长多乱呀。这里的网络处理根本就不应该出现在Controller里面,这样的设计是累赘的,高内聚的,不符合我们程序的设计理念。
所以我们来写写实际需求的代码,现在假设我们要从网络上获取一个json,里面包含一个数组的字符串。我们如何来实现:首先我们需要用到一个model转化库Mantle,同样的方法导入。然后我们来定义我们接收的model:CHListObject.h/.m
@interface CHListObject : MTLModel @property (nonatomic, assign) NSInteger ID;
@property (nonatomic, strong) NSString *name;
@end
#import "CHListObject.h"@implementation CHListObject+(NSDictionary *)JSONKeyPathsByPropertyKey
{
return @{
@"ID":@"id",
@"name":@"name",
};
}@end
我们将在我们的实际获取代码里面把json数据通过这个第三方库转换过来,这是十分方便的。大家同样需要去学一下Mantle的使用方法。(ps:是不是觉得要学很多东西,不要气馁)接下来是我们的获取文件类CHServiceNetRequest.h/.m
#import
#import "CHListObject.h"typedef void(^Failure) (NSError* error);
@interface CHServiceNetRequest : NSObject+(CHServiceNetRequest*)Instance;
/**
*获取列表信息
*
*@paramtoken
*@paramsuccess
*@paramfailure
*/+(void)getListWithToken:(NSString*)token
success:(void(^)(NSArray *list))success
failure:(Failure)failure;
@end
#import "CHServiceNetRequest.h"
#import "CHHttpRequest.h"@implementation CHServiceNetRequest//单例模式
+(CHServiceNetRequest*)Instance
{
static dispatch_once_t once_token;
static CHServiceNetRequest* service = nil;
if (service == nil) {
dispatch_once(&once_token,^{
service = [[CHServiceNetRequest alloc]init];
});
}
return service;
}//需求下的网络请求,返回CHListObject的数组
+(void)getListWithToken:(NSString *)token success:(void (^)(NSArray *))success failure:(Failure)failure
{
NSMutableDictionary *dic = [NSMutableDictionary dictionary];
[dic setObject:token forKey:@"token"];
//==token是什么东西,大家需要自己去了解一下==[CHHttpRequest postWithURLString:@"这里写入一个后台地址" parameters:dic success:^(id responerObject) {
//请求成功
//如果和后台沟通正确,这里的responerObject应该是一个json
//下面开始使用Mantle来处理获取的json
NSArray* data = https://www.it610.com/article/(NSArray*)responerObject;
//由于我们假设json里面是一个数组,所以我们用一个listArr来接收我们的网络数据
NSMutableArray* listArr = [NSMutableArray array];
//取出数据
for (NSDictionary *dic in data) {
CHListObject *obj = [MTLJSONAdapter modelOfClass:[CHListObject class] fromJSONDictionary:dic error:nil];
[listArr addObject:obj];
}
//success输出
if (success) {
success([listArr copy]);
}} failure:^(NSError *error) {
//请求失败
failure(error);
}];
}@end
先在这里告诉大家,这样已经ok了!实际使用时只需要使用CHServiceNetRequest里面的方法就行,具体在那个Controller里使用,直接返回我们json解析出来的listArr数据,Controller里面完全摒弃冗长的代码量。
相信大家可能会跳着看以上的代码,可能也是感觉太长了,难以理解,所以有点放弃了。这里也不多说,相信大家还是能找到适合自己学习的方法的。bye!
【[10分钟教你]封装iOS网络服务】联系方式:401518215@qq.com
推荐阅读
- 移动端|微信小程序组件封装-如何进行属性判断
- 手把手教你用Java获取IP归属地
- 百度工程师教你玩转设计模式(工厂模式)
- Flutter学习LogUtil封装与实现实例详解
- javascript|uni-app 公共请求方法封装
- vue|uni-app网络请求封装(完整版)
- vue|vue.cli项目封装全局axios,请求封装,封装公共的api和调用请求
- javascript|Vue3+vben admin后台管理系统接口封装
- axios组件封装|手把手讲解Vue + Element UI实现后台管理系统(三)(常用模块引入及组件封装(axios数据请求接口封装))
- windows|NSSM - 将任何exe应用封装成windows服务的神器