[10分钟教你]封装iOS网络服务

联系方式:[email protected]

童稚们,我有回来了> <。感觉上次写的教程反响还是挺好的,我动力十足呀。今天我也来带给大家在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请求
 *
 *   @param  URLString  请求的网址字符串
 *   @param  parameters 请求的参数
 *   @param  success    请求成功的回调
 *   @param  failure    请求失败的回调
 */

+(void)getWithURLString:(NSString*)URLString
             parameters:(id)parameters
                success:(void(^)(id responerObject))success
                failure:(void(^)(NSError* error))failure;


/**
 *   发送post请求
 *
 *   @param  URLString  请求的网址字符串
 *   @param  parameters 请求的参数
 *   @param  success    请求成功的回调
 *   @param  failure    请求失败的回调
 */
+(void)postWithURLString:(NSString*)URLString
              parameters:(id)parameters
                 success:(void(^)(id responerObject))success
                 failure:(void(^)(NSError *error))failure;

/**
 *   发送网络请求
 *
 *   @param  URLString  请求的网址字符串
 *   @param  parameters 请求的参数
 *   @param  type       请求的类型
 *   @param  success    请求成功的回调
 *   @param  failure    请求失败的回调
 */

+(void)requestWithURLString:(NSString*)URLString
                 parameters:(id)parameters
                       type:(HttpRequestType)type
                    success:(void(^)(id responseObject))success
                    failure:(void(^)(NSError* error))failure;


/**
 *   上传图片
 *
 *   @param  URLString  请求的网址字符串
 *   @param  parameters 请求的参数
 *   @param  uploadParam上传图片的信息
 *   @param  success    请求成功的回调
 *   @param  failure    请求失败的回调
 */

+(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;

/**
 *    获取列表信息
 *
 *    @param  token
 *    @param  success
 *    @param  failure
 */

+(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 = (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!

联系方式:[email protected]

你可能感兴趣的:([10分钟教你]封装iOS网络服务)