Skip to content

a simple but powerful iOS、macOS networking framework.

License

Notifications You must be signed in to change notification settings

debugly/SCNetworkKit

Repository files navigation

SCNetworkKit

CI Status Version License Platform

SCNetworkKit 是一个功能强大的网络库,支持 iOS / macOS 平台。在编写该库时参考了 MKNetworkKitAFNetworkingMasonryASIHTTPRequest 等优秀开源项目架构的思想,结合了公司项目的实际情况进行逐步改造演变而来。

  • 使用 Objective-C 语言编写
  • 底层封装了 NSURLSession,最低支持 iOS 11.0 / OS X 10.11
  • 采用 Service + Request 分工模式 (从 MKNetworkKit 学习而来)
  • 采用策略模式可配置响应解析器,可以将数据异步解析为 JSON,Model等对象,其中 Model 解析这块算是对 AFNetworking 响应解析模块学习的一个升华,按照自己的思路去完成的
  • 支持了链式编程,实质上是把 block 当做返回值实现的 (从 Masonry 学习而来)
  • 采用 Maker 方式精简对外公开的API长度,使用更方便 (从 Masonry 学习而来)
  • 自创自动取消机制,可将网络请求对象绑定到 x 对象上,当 x 销毁时将自动取消已经发起的网络请求(x 通常是 ViewController)
  • 请求完成,进度回调等完全 Block 化,不支持代理(个人偏爱 Block)
  • 基类支持基础的 GET 和没有 Body 的HTTP请求,带有 body 的 POST 请求则使用子类完成(从 ASIHTTPRequest 学习而来)
  • POST 子类支持了 HTTPBodyStream,轻松搞定大文件上传;弥补了 MKNetworkKit 的一个缺憾
  • 下载类支持断点续传,并且处理了 404 等特殊情况,不会将 404 时服务器响应的错误数据写入文件

SCNetworkKit 演变过程

2016 年我转向 iOS 平台 SDK 相关开发工作,为确保提供出去的 SDK 足够专业,容易集成,防止出现类冲突等报错问题,所以尽量不去依赖主流的开源项目!我造的第一个轮子就是网络请求库,必须能够为 SDK 提供可靠的网络服务,这个库的命名演变过程如下:

SVPNetworkKit -> SLNetworkKit -> SCNetworkKit

  • SVPNetworkKit : 在转向 SDK 开发之前为原项目的上传模块写的一个独立的网络请求模块,坦白讲当时看中了 MKNetworkKit 的精简,并且简单修改之后能够符合上传业务的需求。
  • SLNetworkKit : 转向 SDK 开发工作后,初期只有我一个人,由于 SDK 从 0 到 1,需要做很多基础支持和业务编写,工期十分的紧张,因此将 SVPNetworkKit 直接改名为 SLNetworkKit,然后在此基础上进行修改。这个阶段主要支持了 Maker 形式的调用方式、抽取了响应解析模块、支持了Model解析、响应异步解析、自动取消等机制。
  • SCNetworkKit : 随着 SDK 业务的增多,并且要求能够灵活选取某几个组合集成!因此迫切需要将原来 SDK 中的基础库进行下沉,形成一个更加通用的SC底层库,所有的 SDK 均依赖于该库。SLNetworkKit 便是其中之一,顺势改其前缀为 SC !这个阶段主要支持了 Stream Body 的 POST 请求,方便大文件上传!于 2017 年开源。

目录结构

├── Example
│   ├── Server
│   └── SCNetworkDemo
├── LICENSE
├── README.md
├── SCNetworkKit
│   └── Classes
├── SCNetworkKit.podspec
└── _config.yml
  • Example/SCNetworkDemo : 包含了 iOS、macOS 平台配套调用示例
  • SCNetworkKit/Classes : 源码
  • Example/Server : 使用 Express 编写的简单 Node 服务器,主要为 Demo 提供 GET/POST 请求测试支持,客户端上传的文件都放在 Server/upload 文件夹下面。

Server 使用方法:

cd Server
//第一次运行需要安装下依赖库,以后执行就不用了
npm install
//启动 server
npm start

安装方式

  • 使用 CocoaPods 安装

    source 'https://github.com/CocoaPods/Specs.git'
    platform :ios, '11.0'
    
    target 'TargetName' do
    pod 'SCNetworkKit'
    end
    
  • 使用源码

    下载最新 release 代码,找到 SCNetworkKit 目录,拖到工程里即可。

使用范例

假设服务器返回的数据格式如下:

{ 
  code = 0;
  content =     {
     entrance =         (
         {
             isFlagship = 0;
             name = "\U65f6\U5c1a\U6f6e\U65f6\U5c1a";
             pic = "http://pic12.shangpin.com/e/s/15/03/03/20150303151320537363-10-10.jpg";
             refContent = "http://m.shangpin.com/meet/185";
             type = 5;
         },
         {
            //....
         }
       )
     }
 }

下面演示如何通过配置不同的解析器,从而达到着陆 block 回调不同结果的效果:

  • 发送 GET请求,回调原始 Data,不做解析

    SCNetworkRequest *req = [[SCNetworkRequest alloc]initWithURLString:kTestJSONApi params:nil];
    ///因为默认解析器是SCNJSONResponseParser;会解析成JSON对象;所以这里不指定解析器,让框架返回data!
    req.responseParser = nil;
    [req addCompletionHandler:^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(result,err);
        }
    }];
    
    [[SCNetworkService sharedService]startRequest:req];
  • 发送 GET请求,回调 JOSN 对象

    SCNJSONResponseParser *responseParser = [SCNJSONResponseParser parser];
    ///框架会检查接口返回的 code 是不是 0 ,如果不是 0 ,那么返回给你一个err,并且result是 nil;
    responseParser.checkKeyPath = @"code";
    responseParser.okValue = @"0";
    
    ///support chain
    SCNetworkRequest *req = [[SCNetworkRequest alloc]init];
    
    req
    .c_URL(kTestJSONApi)
    .c_ResponseParser(responseParser)
    .c_CompletionHandler(^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(result,err);
        }
    });
    [[SCNetworkService sharedService]startRequest:req];
  • 发送 GET 请求,回调 Model 对象

    SCNetworkRequest *req = [[SCNetworkRequest alloc]initWithURLString:kTestJSONApi params:nil];
    
    SCNModelResponseParser *responseParser = [SCNModelResponseParser parser];
    ///解析前会检查下JSON是否正确;
    responseParser.checkKeyPath = @"code";
    responseParser.okValue = @"0";
    ///根据服务器返回数据的格式和想要解析结构对应的Model配置解析器
    responseParser.modelName = @"TestModel";
    responseParser.targetKeyPath = @"content/entrance";
    req.responseParser = responseParser;
    [req addCompletionHandler:^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(result,err);
        }
    }];
    [[SCNetworkService sharedService]startRequest:req];

由于上面有 JSON 转 Model 的过程,因此在使用之前需要注册一个对应的解析器,你可以到 demo 里搜下 [SCNModelResponseParser registerModelParser:[SCNModelParser class]]; 具体看下究竟。

  • 文件下载

    SCNetworkRequest *get = [[SCNetworkRequest alloc]initWithURLString:kTestDownloadApi2 params:nil];
    //NSString *path = [NSTemporaryDirectory()stringByAppendingPathComponent:@"node.jpg"];
    NSString *path = [NSTemporaryDirectory()stringByAppendingPathComponent:@"test.mp4"];
    NSLog(@"download path:%@",path);
    get.downloadFileTargetPath = path;
    get.responseParser = nil;
    [get addCompletionHandler:^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(path,err);
        }
    }];
    
    [get addProgressChangedHandler:^(SCNetworkRequest *request, int64_t thisTransfered, int64_t totalBytesTransfered, int64_t totalBytesExpected) {
    
        if (totalBytesExpected > 0) {
            float p = 1.0 * totalBytesTransfered / totalBytesExpected;
            NSLog(@"download progress:%0.4f",p);
            if (progress) {
                progress(p);
            }
        }
    }];
    
    [[SCNetworkService sharedService]startRequest:get];
  • 文件上传

    NSDictionary *ps = @{@"name":@"Matt Reach",@"k1":@"v1",@"k2":@"v2",@"date":[[NSDate new]description]};
    SCNetworkPostRequest *post = [[SCNetworkPostRequest alloc]initWithURLString:kTestUploadApi params:ps];
    
    SCNetworkFormFilePart *filePart = [SCNetworkFormFilePart new];
    NSString *fileURL = [[NSBundle mainBundle]pathForResource:@"logo" ofType:@"png"];
    filePart.data = [[NSData alloc]initWithContentsOfFile:fileURL];
    filePart.fileName = @"logo.png";
    filePart.mime = @"image/jpg";
    filePart.name = @"logo";
    
    SCNetworkFormFilePart *filePart2 = [SCNetworkFormFilePart new];
    filePart2.fileURL = [[NSBundle mainBundle]pathForResource:@"node" ofType:@"txt"];
    
    post.formFileParts = @[filePart,filePart2];
    [post addCompletionHandler:^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(result,err);
        }
    }];
    
    [post addProgressChangedHandler:^(SCNetworkRequest *request, int64_t thisTransfered, int64_t totalBytesTransfered, int64_t totalBytesExpected) {
    
        if (totalBytesExpected > 0) {
            float p = 1.0 * totalBytesTransfered / totalBytesExpected;
            NSLog(@"upload progress:%0.4f",p);
            if (progress) {
                progress(p);
            }
        }
    }];
    
    [[SCNetworkService sharedService]startRequest:post];
  • 通过表单POST数据

    NSDictionary *ps = @{@"name":@"Matt Reach",@"k1":@"v1",@"k2":@"v2",@"date":[[NSDate new]description]};
    SCNetworkPostRequest *post = [[SCNetworkPostRequest alloc]initWithURLString:kTestUploadApi params:ps];
    post.parameterEncoding = SCNPostDataEncodingFormData;
    [post addCompletionHandler:^(SCNetworkRequest *request, id result, NSError *err) {
    
        if (completion) {
            completion(result,err);
        }
    }];
    
    [[SCNetworkService sharedService]startRequest:post];

链式编程

SCNJSONResponseParser *responseParser = [SCNJSONResponseParser parser];
///框架会检查接口返回的 code 是不是 0 ,如果不是 0 ,那么返回给你一个err,并且result是 nil;
responseParser.checkKeyPath = @"code";
responseParser.okValue = @"0";

///support chain

SCNetworkRequest *req = [[SCNetworkRequest alloc]init];

req.c_URL(@"http://debugly.cn/dist/json/test.json")
   .c_ResponseParser(responseParser);
   .c_CompletionHandler(^(SCNetworkRequest *request, id result, NSError *err) {

            if (completion) {
                completion(result);
            }
       });
[[SCNetworkService sharedService]sendRequest:req];

架构设计

  • 综合参考了 MKNetwork2.0 和 AFNetwork 2.0 的设计,吸取了他们的精华,去掉了冗余的设计,融入了自己的想法,将网络请求抽象为 Request 对象,并由 Service 管理,Service 为 Request 分配代理对象 --- 处理传输数据、请求结束,请求失败等事件,请求结束后通过改变 Rquest 的 state 属性,告知 Request 请求结束,然后根据配置的响应解析器,异步解析数据,结果可能是 data, string, json, model, image 等等;最终通过我们添加到 Request 对象上的 completionBlock 回调给调用层。

设计图:

采用注册的方式解耦和

功能强大的同时要顾及到扩展性,本框架支持很多扩展,以响应解析为例,你可以继续创建你想要的解析器;可以使用你喜欢的 JOSN 转 Model 框架来做解析;可以让网络库解析更多格式的图片;这些都是可以做到的,并且还很简单。

  • 由于框架配备了支持 JSON 转 Model 的 SCNModelResponseParser 响应解析器,那么就不得不依赖于 JSON 转 Model 的框架,考虑到项目中很可能已经有了这样的框架,因此并没有将这块逻辑写死,而是采用注册的方式,来扩展 SCN 的能力!所以使用 Model 解析器之前必须注册 一个用于将 JOSN 转为 Model 的类,该类实现 SCNModelParserProtocol 协议!为了方便,最好是在APP启动后就注册,或者创建 Service 的时候创建,以免使用的时候还没注册,导致崩溃!

    @protocol SCNModelParserProtocol <NSObject>
    
    @required;
    + (id)JSON2Model:(id)json modelName:(NSString *)mName;
    
    @end
    
    @interface SCNModelResponseParser : SCNJSONResponseParser
    
    @property (nonatomic,copy) NSString *modelName;
    
    + (void)registerModelParser:(Class<SCNModelParserProtocol>)parser;
    
    @end

我在 demo 里面使用的是我的另外一个轮子:SCJSONUtil ;具体实现可查看demo。

  • 图片解析器默认支持 png、jpg 图片格式,当下 webp 格式由于体积更小,很多厂商开始使用,我的 SDK 里也用到了这个格式,因此我在 SDK 里注册了解析 webp 的解析类;

    @protocol SCNImageParserProtocol <NSObject>
    
    @required;
    + (UIImage *)imageWithData:(NSData *)data scale:(CGFloat)scale;
    
    @end
    
    ///默认支持png 和 jpg;可通过注册的方式扩展!
    @interface SCNImageResponseParser : SCNHTTPResponseParser

    /** 注册新的图片解析器和对应的类型

    @param parser 解析器 @param mime 支持的类型 */

    • (void)registerParser:(Class)parser forMime:(NSString *)mime;

      @end

      
      

这种注册器的方式优雅地扩充了网络库的功能,就好比插件一样,插上就能用,只需要规格上符合我协议里规定的要求即可!反之,如果你不需要解析 webp, 你不需要 json 转 model 的话,你就没必要去插对应的模块!

如果没有注册器这么一个好的实践的话,要达到同样的扩展效果可能就很难了!如果你有别的点子请联系我。

SCNetworkService

由上图可知,SCNetworkService 主要起到了发起网络请求,处理好 Request,task,delegate 对象的一一对应关系的作用!

  • 为了方便使用,还提供了可用于整个 App 的共享 SCNetworkService 对象,用来发送普通的网络请求;当然你有必要为不同的业务创建不同的 Service;一个 Service 内部则对应了一个 NSURLSession 对象!

SCNetworkRequest

NSURLSession 管理的网络请求结束后,会在 SCNetworkRequest 里处理响应数据,根据配置的 ResponseParser 去异步解析,最终在主线程里安全着陆;

  • SCNetworkRequest 从 start 开始被 Service 持有,直到着陆后 Service 不再持有,因此上层可以不持有 SCNetworkRequest 对象!如果要拥有 SCNetworkRequest 对象的指针,一般使用 weak 即可;
  • SCNetworkRequest 支持添加多个回调,回调顺序跟添加的顺序一样;
  • 注意添加回调的时候,不要让 SCNetworkRequest 持有你的对象,否则 SCNetworkRequest 会一直持有,直到着陆,虽然不会导致循环引用导致的内存泄漏,但是却“延长”了被持有对象的生命周期;
  • 该类默认发送 GET 请求,也可以修改 method 发送 POST 请求,但是只能发送不带 body 体的 POST 请求。

SCNetworkPostRequest

继承了 SCNetworkRequest,专门用于发送带有 body 体的 POST 请求,body 体内容支持四种编码方式:

  • SCNPostDataEncodingURL : application/x-www-form-urlencoded;
  • SCNPostDataEncodingJSON : application/json;
  • SCNPostDataEncodingPlist : application/x-plist;
  • SCNPostDataEncodingFormData : multipart/form-data;

只有使用 SCNPostDataEncodingFormData 方式的请求会采用 HTTPBodyStream !

版本

  • 1.0.5 : 支持 stream HTTPBody,轻松搞定大文件上传
  • 1.0.6 : 支持一次上传多个文件,配套 Node 上传文件服务器
  • 1.0.7 : 修复直接使用二进制上传失败问题(重复计算长度,导致Content Length计算偏大)
  • 1.0.8 : 支持 macOS 平台 (暂不支持图片解析)
  • 1.0.9 : 整理目录,POST 请求可添加 Query 参数
  • 1.0.10 : 修改默认 UA 格式
  • 1.0.11 : 抽取解析过程,可完全自定义;支持 JSONUtil 的动态映射
  • 1.0.12 : 解除并发数为 2 的限制,使用系统配置
  • 1.0.13 : 支持自定义请求body体
  • 1.0.14 : 修改默认UA
  • 1.0.15 : 下载文件支持断点续传
  • 1.0.16 : 将下载逻辑抽取为单独的类
  • 1.0.17 : 移除自动取消支持,这一功能抽取为了单独的 MRDeallocSubscriber 模块,可通过block形式完成,使得网络库更加纯粹
  • 1.0.18 : 整理下载文件逻辑,使用单独的属性控制断点续传,避免404等情况的响应数据写入文件
  • 1.0.19:基类支持发送不带 body 体的 POST 请求
  • 1.0.20:修复断点续传 Range 请求 416 问题
  • 1.0.21:修复下载类重写 getter 导致的调用堆栈溢出问题
  • 1.0.22:修复使用Model解析默认将 JOSN 里的值转成 String 问题
  • 1.0.23: 下载文件时自动创建父级目录,避免下载出错
  • 1.0.24: 支持通过 NSURLRequest 初始化请求对象
  • 1.0.25: 修复参数字典为空时会在请求后面拼接?或者&的问题
  • 1.0.26: 支持下载限速功能
  • 1.0.27: 遇到错误时,userinfo的statusCode字段使用HTTP状态码
  • 1.0.28: 避免写文件可能引发的崩溃
  • 1.0.29: 写入文件失败时,回调错误状态
  • 1.0.30: 下载支持限速
  • 1.0.31: JSON响应解析器默认包含 application/octet-stream 类型
  • 1.0.32: 支持 tvOS 平台
  • 1.0.33: 取消时清理相关的block,避免持有捕获对象

由于该网络库是完全为自己业务服务的,因此不是所有的功能都很完善,而是用到时再加,发现不合理就改,所以如果你使用了 SCNetworkKit ,发现功能缺失,可以提交 PR 或者 Issue 给我!

About

a simple but powerful iOS、macOS networking framework.

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •