AFNetWorking分析<二>

AFHTTPSessionManager

通常我们在运用AFN框架进行网络请求时,使用的都是AFHTTPSessionManager这个类。AFHTTPSessionManager继承于AFURLSessionManager,是对网络请求的进一步封装。这个类将繁琐的配置request、拼接formdata等工作进行了封装,仅仅提供GETPOSTHEADPUTDELETE这几个非常方便直观的API。
AFHTTPSessionManager相对于其父类,新添加了三个属性baseURLrequestSerializerresponseSerializer

请求器
@property (nonatomic, strong) AFHTTPRequestSerializer <AFURLRequestSerialization> * requestSerializer;

AFHTTPRequestSerializer这个类就是AFN框架对于网络请求中request配置的封装,它服从AFURLRequestSerialization协议,之后会详细讲解。在使用AFHTTPSessionManager时候,这个属性是不能为nil的,在它的init方法中,是初始化为[AFHTTPRequestSerializer serializer],当然也可以自己改变这个请求器,这个取决于你的后台要接受什么类型的数据,如果你的后台是要接收json格式的请求那么就是[AFJSONRequestSerializer serializer]

响应器
@property (nonatomic, strong) AFHTTPResponseSerializer <AFURLResponseSerialization> * responseSerializer;

AFHTTPResponseSerializer这个类是对网络请求响应的封装,通过改变这个属性,AFN框架可以自动对请求下来的数据进行解析。例如,你请求下来的数据是json格式,那么将responseSerializer赋值成[AFJSONResponseSerializer serializer],于是你得到就是解析后的数据。这里要注意,如果在请求时出现3840的错误码,那就是你的responseSerializer有问题,很有可能请求下来的不是json串,而你指定它要json解析。
因为我们平常开发常用到请求方式一般是两种:GETPOST。所以,我就以这两个请求方式为例。
通常我们使用+ (instancetype)manager类方法,以此调用初始化方法

- (instancetype)initWithBaseURL:(NSURL *)url
           sessionConfiguration:(NSURLSessionConfiguration *)configuration
{
    self = [super initWithSessionConfiguration:configuration];
    if (!self) {
        return nil;
    }

    // Ensure terminal slash for baseURL path, so that NSURL +URLWithString:relativeToURL: works as expected
    if ([[url path] length] > 0 && ![[url absoluteString] hasSuffix:@"/"]) {
        url = [url URLByAppendingPathComponent:@""];
    }

    self.baseURL = url;

    self.requestSerializer = [AFHTTPRequestSerializer serializer];
    self.responseSerializer = [AFJSONResponseSerializer serializer];

    return self;
}

在这里看到init方法,对请求器与响应器进行了初始化赋值。
之后我们会调用例如下面的方法

GET方法
- (NSURLSessionDataTask *)GET:(NSString *)URLString
                   parameters:(id)parameters
                      success:(void (^)(NSURLSessionDataTask *task, id responseObject))success
                      failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure
{
    NSURLSessionDataTask *dataTask = [self dataTaskWithHTTPMethod:@"GET" URLString:URLString parameters:parameters success:success failure:failure];

    [dataTask resume];

    return dataTask;
}

在这个方法内,调用

NSURLSessionDataTask *dataTask = [self dataTaskWithHTTPMethod:@"GET" URLString:URLString parameters:parameters success:success failure:failure];

得到dataTask之后执行resume方法。在这个方法里,首先会根据我们进行网络请求的方法来配置request。这时就用到了AFHTTPSessionManagerrequestSerializer属性,通过属性中的值来调用类的实例方法,之后,判断是否配置错误,如果配置错误,则通过GCD在completionQueue这个队列中会调出错误信息,这里如果你没有对这个属性进行赋值的话,它会选择在主线程回调错误信息。配置好request之后就调用父类的网络请求的方法,得到dataTask返回,在请求完成时,执行success或者failure的block。注意,这里得到的dataTask需要回调给外部,所以需要__block修饰。

- (NSURLSessionDataTask *)dataTaskWithHTTPMethod:(NSString *)method
                                       URLString:(NSString *)URLString
                                      parameters:(id)parameters
                                         success:(void (^)(NSURLSessionDataTask *, id))success
                                         failure:(void (^)(NSURLSessionDataTask *, NSError *))failure
{
    NSError *serializationError = nil;
    NSMutableURLRequest *request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:URLString relativeToURL:self.baseURL] absoluteString] parameters:parameters error:&serializationError];
    if (serializationError) {
        if (failure) {
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wgnu"
            dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{
                failure(nil, serializationError);
            });
#pragma clang diagnostic pop
        }

        return nil;
    }

    __block NSURLSessionDataTask *dataTask = nil;
    dataTask = [self dataTaskWithRequest:request completionHandler:^(NSURLResponse * __unused response, id responseObject, NSError *error) {
        if (error) {
            if (failure) {
                failure(dataTask, error);
            }
        } else {
            if (success) {
                success(dataTask, responseObject);
            }
        }
    }];

    return dataTask;
}

在这些提供给外界使用的api里,有一个api是特殊的

- (NSURLSessionDataTask *)POST:(NSString *)URLString
                    parameters:(id)parameters
     constructingBodyWithBlock:(void (^)(id <AFMultipartFormData> formData))block
                       success:(void (^)(NSURLSessionDataTask *task, id responseObject))success
                       failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure

这个方法是我们进行网络上传时使用的方法,我们可以看到它多加入了一个block参数,这个block中有一个服从AFMultipartFormData协议的参数formData,从字面上我们就可以知道,如果我们需要上传什么数据的话只需要往这个参数后面进行拼接就可以了,事实上也的确如此。在这个POST方法中,调用了requestSerializer的另外一个用来配置上传文件的request的方法。

- (NSMutableURLRequest *)multipartFormRequestWithMethod:(NSString *)method
                                              URLString:(NSString *)URLString
                                             parameters:(NSDictionary *)parameters
                              constructingBodyWithBlock:(void (^)(id <AFMultipartFormData> formData))block
                                                  error:(NSError *__autoreleasing *)error

AFURLRequestSerialization中,构建Multipart请求是占篇幅很大的一个功能,它也的确值得耗费更多的代码。在上一章,我已经讲了在iOS设备上传文件时是multipart协议上传,所以需要按照格式进行配置request,这里就不在赘述了。在用NSURLRequest上传文件时,一般是两种方法,一个是设置body,但是如果文件稍大的话,将会撑爆内存。另外一种则是,创建一个临时文件,将数据拼接进去,然后将文件路径设置为bodyStream,这样就可以分片的上传了。而AFN框架则是更进一步的运用边传边拼的方式上传文件,这无疑是更加高端也是更加繁琐的方法。
这里通过constructingBodyWithBlock向使用者提供了一个AFStreamingMultipartFormData对象,调这个对象的append方法, AFStreamingMultipartFormData内部把这些append的数据转成不同类型的AFHTTPBodyPart,添加到自定义的 AFMultipartBodyStream 里。最后把AFMultipartBodyStream赋给原来NSMutableURLRequest的bodyStream。NSURLConnection发送请求时会读取这个 bodyStream,在读取数据时会调用这个 bodyStream 的 -read:maxLength:方法,AFMultipartBodyStream重写了这个方法,不断读取之前 append进来的AFHTTPBodyPart 数据直到读完。

- (NSInteger)read:(uint8_t *)buffer
        maxLength:(NSUInteger)length
{
    if ([self streamStatus] == NSStreamStatusClosed) {
        return 0;
    }

    NSInteger totalNumberOfBytesRead = 0;

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wgnu"
    while ((NSUInteger)totalNumberOfBytesRead < MIN(length, self.numberOfBytesInPacket)) {
        if (!self.currentHTTPBodyPart || ![self.currentHTTPBodyPart hasBytesAvailable]) {
            if (!(self.currentHTTPBodyPart = [self.HTTPBodyPartEnumerator nextObject])) {
                break;
            }
        } else {
            NSUInteger maxLength = length - (NSUInteger)totalNumberOfBytesRead;
            NSInteger numberOfBytesRead = [self.currentHTTPBodyPart read:&buffer[totalNumberOfBytesRead] maxLength:maxLength];
            if (numberOfBytesRead == -1) {
                self.streamError = self.currentHTTPBodyPart.inputStream.streamError;
                break;
            } else {
                totalNumberOfBytesRead += numberOfBytesRead;

                if (self.delay > 0.0f) {
                    [NSThread sleepForTimeInterval:self.delay];
                }
            }
        }
    }
#pragma clang diagnostic pop

    return totalNumberOfBytesRead;
}

下图就是multipart方式进行上传文件的request配置的步骤图。


NSMutableURLRequest的构建步骤

AFMultipartBodyStream

AFMultipartBodyStreamNSInputStream的子类,有人觉得是不是只要简单的将这个类setHTTPBodyStream给request就可以了?事实上并不是这样,用NSURLRequest 发出请求会导致 crash,提示

[xx _scheduleInCFRunLoop:forMode:]: unrecognized selector

这是因为NSURLRequest实际上接受的不是NSInputStream 对象,而是 CoreFoundation 的 CFReadStreamRef 对象,因为 CFReadStreamRefNSInputStream 是 toll-free bridged,可以自由转换,但CFReadStreamRef 会用到 CFStreamScheduleWithRunLoop 这个方法,当它调用到这个方法时,object-c 的 toll-free bridging 机制会调用 object-c 对象 NSInputStream 的相应函数,这里就调用到了_scheduleInCFRunLoop:forMode:,若不实现这个方法就会crash。是不是觉得好绕啊?的确,在学习这套框架的时候,我不停在的感慨大神就是大神,给你一种非常完美的感觉。其实AFN框架绝不仅仅是只有这几个重点,剩下的东西还有很多很多,例如还有AFURLResponseSerialization,和网络请求验证证书的A'FSecurityPolicy。整体的架构真的很漂亮,绝对是iOS开发工程师必需学习研究的著名框架之一。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容