基于AFN的二次封装

一.前言

最近重构项目,遇到了很多问题,也从中总结出了几点经验,这里先讲一讲网络工具类的封装,网络请求是前端和后台沟通的桥梁,那么前端网络工具类的封装或者使用是一个项目的基础。由于本人现做的项目“年久失修、满目疮痍”,项目14年建立之后经历不下于5人之手,每人编程习惯不同,勿论对错,但从网络工具的使用来看就显得非常之乱:有用原生NSURLSession(NSURLSession是苹果在iOS7后为HTTP数据传输提供的一系列接口,比NSURLConnection强大,坑少,好用)、有用原生NSURLConnection(NSURLConnection在iOS9被宣布弃用)、有用ANFNetWorking(本人弃用ASI后就一直在用AFN它是对NSURLSession封装较好作者并持续更新的框架)、还有用Overcoat。为了统一网络请求,方便以后开发,我就封装了网络工具类,此工具类基于AFNetWorking,封装了GET、POST、Upload、Download方法,关于网络监测之前文章有写,在此不做赘述,欢迎留言批评指正!

二.封装思路
  • 创建网络工具类单例,在单例创建的时候配置相关参数,例如:新增一些响应解析器能够接受的数据类型(接受数据不全导致网络请求错误是AFN的一大坑)、设置超时时间、配置请求头(这里我们把access_token验证放到了请求头里,又的项目是放到了请求体或参数里)、配置相应器和解析器等
  • 基于AFHTTPSessionManager(是对NSURLSession的封装)GET、POST、Upload、Download方法二次封装
  • 上传Upload方法需要上传数据的参数,所以这里建立了个参数模型:WKUploadParam
三.源码
  • 网络工具类 WKNetWorkTool.h
      /*
       **此网络工具类是LWK新建网络工具类
        *为统一管理降低耦合新建模块均用此工具类
        *
        *
        */
    
    #import <Foundation/Foundation.h>
    #import <AFNetworking/AFHTTPSessionManager.h>
    #import "WKUploadParam.h"
    @interface WKNetWorkTool : AFHTTPSessionManager
    
      /**
        *  二次封装网络工具单例
        *
        */
    + (instancetype)sharedTool;
    
       /**
         *  基于AFN二次封装GET方法
         *
         *  @URLString    相对路径
         *  @params       请求参数
         *  @finish      完成回调
         *
         */
      - (void)requestGET:(NSString *)URLString parames:(id)parames success:(void (^)(id responseObj))success failure:(void (^)(id error))failure;
    
       /**
         *   基于AFN二次封装POST方法
         *
         *  @URLString    相对路径
         *  @params       请求参数
         *  @finish      完成回调
         */
    - (void)requestPOST:(NSString *)URLString parames:(id)parames success:(void (^)(id responseObj))success failure:(void (^)(id error))failure;
         /**
           *  上传图片 (单张或一组)
           *
           *  @param URLString   上传图片的网址字符串
           *  @param parameters  上传图片的参数
           *  @param uploadParam 上传图片的信息
           *  @param success     上传成功的回调
           *  @param failure     上传失败的回调
           */
    - (void)uploadWithURLString:(NSString *)URLString
               parameters:(id)parameters
              uploadParam:(NSArray <WKUploadParam *> *)uploadParams
                  success:(void (^)())success
                  failure:(void (^)(NSError *error))failure;
         /**
          *  下载数据
          *
          *  @param URLString   下载数据的网址
          *  @param parameters  下载数据的参数
          *  @param success     下载成功的回调
          *  @param failure     下载失败的回调
           */
    - (void)downLoadWithURLString:(NSString *)URLString
                 parameters:(id)parameters
                   progerss:(void (^)())progress
                    success:(void (^)())success
                    failure:(void (^)(NSError *error))failure;
    
    @end
    
  • 网络工具类 WKNetWorkTool.m
    #import "WKNetWorkTool.h"
    #import "SDHSDataCache.h"
    @implementation WKNetWorkTool
    
    #pragma mark - 单例
    
    + (instancetype)sharedTool{
         static dispatch_once_t onceToken;
         static WKNetWorkTool *instance;
         dispatch_once(&onceToken, ^{
      instance = [[super alloc]init];
      instance.responseSerializer = [AFJSONResponseSerializer serializer];
      AFHTTPRequestSerializer *requestSerializer = [AFHTTPRequestSerializer serializer];
      requestSerializer.timeoutInterval = 10;
      ///我们项目是把access_token(后台验证用户省份标识)放在了请求头里,有的项目是放在了请求体里,视实际情况而定
      [requestSerializer setValue:[SDHSDataCache getToken] forHTTPHeaderField:@"access_token"];
      instance.requestSerializer = requestSerializer;
      
      //        ///1.强制更换AFN数据解析类型 只支持一下添加的数据类型 AFN自带的就没有了 如果AFN新增了数据解析类型 这里也没有变化 所以有下面2方法 向原有可解析数据类型添加较好
      //        instance.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/html",@"plant/html",nil];8k9K10
      
      ///2.获取AFN原有的数据解析类型 然后新增一些响应解析器能够接受的数据类型
      NSMutableSet *acceptableContentTypes = [NSMutableSet setWithSet:instance.responseSerializer.acceptableContentTypes];
      [acceptableContentTypes addObjectsFromArray:@[@"application/json", @"text/json", @"text/javascript",@"text/html",@"plant/html",@"text/plain",@"text/xml"]];
      instance.responseSerializer.acceptableContentTypes = acceptableContentTypes;
      
    });
     return instance;
    }
    
    #pragma mark - 自定义GET
    
    - (void)requestGET:(NSString *)URLString parames:(id)parames success:(void (^)(id responseObj))success failure:(void (^)(id error))failure{
       //AFN没有做UTF8转码 防止URL字符串中含有中文或特殊字符发生崩溃
         URLString = [[NSString  stringWithFormat:@"%@%@",nstrPublicUrl,URLString]stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
         [self GET:URLString parameters:parames success:^(NSURLSessionDataTask * _Nonnull task, id  _Nonnull responseObject) {
      success(responseObject);
           } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
      failure(error);
      }];
    
    }
    
     #pragma mark - 自定义POST
    
    - (void)requestPOST:(NSString *)URLString parames:(id)parames success:(void (^)(id responseObj))success failure:(void (^)(id error))failure{
       //AFN没有做UTF8转码 防止URL字符串中含有中文或特殊字符 发生崩溃
          URLString = [[NSString stringWithFormat:@"%@%@",nstrPublicUrl,URLString]stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
          [self POST:URLString parameters:parames success:^(NSURLSessionDataTask * _Nonnull task, id  _Nonnull responseObject) {
      
      ////将接收回来的数据转成UTF8的字符串,然后取出格式占位符 加上个转义符后才能让数据进行转换 否则转换失败
      //        NSString*jsonString = [[NSString alloc] initWithBytes:[responseObject bytes]length:[responseObject length]encoding:NSUTF8StringEncoding];
      //        jsonString = [jsonString stringByReplacingOccurrencesOfString:@"\t" withString:@"\\t"];
      //        NSData * jsonData = [jsonString dataUsingEncoding:NSUTF8StringEncoding];
      
      success(responseObject);
        } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
      failure(error);
       }];
     }
    
    #pragma mark - 上传数据
    
      - (void)uploadWithURLString:(NSString *)URLString parameters:(id)parameters uploadParam:(NSArray<WKUploadParam *> *)uploadParams success:(void (^)())success failure:(void (^)(NSError *))failure {
               URLString = [NSString stringWithFormat:@"%@%@",nstrPublicUrl,URLString];
               [self POST:URLString parameters:parameters constructingBodyWithBlock:^(id<AFMultipartFormData>  _Nonnull formData) {
      for (WKUploadParam *uploadParam in uploadParams) {
          [formData appendPartWithFileData:uploadParam.data name:uploadParam.name fileName:uploadParam.filename mimeType:uploadParam.mimeType];
      }
            } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nonnull responseObject) {
      if (success) {
          success(responseObject);
      }
         } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
      if (failure) {
          failure(error);
      }
       }];
     }
    
    #pragma mark - 下载数据
    
     - (void)downLoadWithURLString:(NSString *)URLString parameters:(id)parameters progerss:(void (^)())progress success:(void (^)())success failure:(void (^)(NSError *))failure {
              URLString = [NSString stringWithFormat:@"%@%@",nstrPublicUrl,URLString];
              NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:URLString]];
              NSURLSessionDownloadTask *downLoadTask = [self downloadTaskWithRequest:request progress:nil destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) {
      return targetPath;
         } completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nullable filePath, NSError * _Nullable error) {
      if (failure) {
          failure(error);
      }
         }];
             [downLoadTask resume];
           }
    
    @end
    
  • 上传参数模型 WKUploadParam.h
       #import <Foundation/Foundation.h>
    
       @interface WKUploadParam : NSObject
        /**
           *  图片的二进制数据
          */
       @property (nonatomic, strong) NSData *data;
        /**
          *  服务器对应的参数名称
          */
       @property (nonatomic, copy) NSString *name;
        /**
          *  文件的名称(上传到服务器后,服务器保存的文件名)
          */
       @property (nonatomic, copy) NSString *filename;
        /**
          *  文件的MIME类型(image/png,image/jpg等)
          */
       @property (nonatomic, copy) NSString *mimeType;
       @end
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容