AFNetworking源码简读

AFNetworking源码简读_第1张图片
image

一。首先我们用AFNetworking请求一下百度首页:

-(void)loadSth{
    AFHTTPSessionManager *manager = [[AFHTTPSessionManager alloc]init];
   [manager GET:@"https://www.baidu.com" parameters:nil progress:nil   success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
    NSLog(@"OK");
 } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
     NSLog(@"Fail");
  }];
 }

这个请求结果是失败的,打印error如下:
Error Domain=com.alamofire.error.serialization.response Code=-1016 "Request failed: unacceptable content-type: text/html" UserInfo={NSLocalizedDescription=Request failed: unacceptable content-type: text/html,......}

为什么会失败?

这是因为 AFNetworking默认把响应结果当成json来处理,(默认manager.responseSerializer = [AFJSONResponseSerializer serializer]) ,很显然,我们请求的百度首页 返回的并不是一个json文本,而是一个html网页,但是AFNetworking并不知道,它坚信请求的结果就是一个json文本!然后固执地以json的形式去解析,显然没办法把一个网页解析成一个字典或者数组,所以产生了上述错误。
然而,我们期望它能够正确地处理这个情形,而不是提示一个错误.
这时候 你必须告诉AFNetworking:别把这个网页当json来处理!
只需要在发送请求前加入:manager.responseSerializer = [AFHTTPResponseSerializer serializer];
这样就会成功了!成功之后返回的responseObject,是一个NSData对象,把这个对象转换成NSString对象输出,就是一个HTML的文本,对应百度首页。
接下来我们会看到AFHTTPResponseSerializer,AFJSONResponseSerializer的相关介绍

二。接下来再看看AFHTTPSessionManager的代码

+ (instancetype)manager {
    return [[[self class] alloc] initWithBaseURL:nil];
}

类方法,用来生成AFHTTPSessionManager对象。
接下来就是就是一系列初始化方法,最后都会调用

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

这这个方法中还创建了AFHTTPRequestSerializer对象requestSerializer,这个

三。AFURLSessionManager

是AFHTTPSessionManager的基类,上面的AFHTTPSessionManager的初始化方法,调用了基类的初始化方法:

- (instancetype)initWithSessionConfiguration:(NSURLSessionConfiguration *)configuration {
    self = [super init];
    if (!self) {
        return nil;
    }
    if (!configuration) {
        configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
    }
    self.sessionConfiguration = configuration;
    self.operationQueue = [[NSOperationQueue alloc] init];
    self.operationQueue.maxConcurrentOperationCount = 1;
    self.session = [NSURLSession sessionWithConfiguration:self.sessionConfiguration delegate:self delegateQueue:self.operationQueue];
    self.responseSerializer = [AFJSONResponseSerializer serializer];//基类只有responseSerializer
    self.securityPolicy = [AFSecurityPolicy defaultPolicy];

#if !TARGET_OS_WATCH
    self.reachabilityManager = [AFNetworkReachabilityManager sharedManager];
#endif
    
    self.mutableTaskDelegatesKeyedByTaskIdentifier = [[NSMutableDictionary alloc] init];
    self.lock = [[NSLock alloc] init];
    self.lock.name = AFURLSessionManagerLockName;
    [self.session getTasksWithCompletionHandler:^(NSArray *dataTasks, NSArray *uploadTasks, NSArray *downloadTasks) {
        for (NSURLSessionDataTask *task in dataTasks) {
            [self addDelegateForDataTask:task uploadProgress:nil downloadProgress:nil completionHandler:nil];
        }
        for (NSURLSessionUploadTask *uploadTask in uploadTasks) {
            [self addDelegateForUploadTask:uploadTask progress:nil completionHandler:nil];
        }
        for (NSURLSessionDownloadTask *downloadTask in downloadTasks) {
            [self addDelegateForDownloadTask:downloadTask progress:nil destination:nil completionHandler:nil];
        }
    }];
    return self;
}

部分代码解释

self.session = [NSURLSession sessionWithConfiguration:self.sessionConfiguration delegate:self delegateQueue:self.operationQueue];
创建了NSURLSession类型的成员变量session,这个是网络请求的基础,来自苹果SDK中的类。
同时设置self,也就是AFURLSessionManager的对象作为session的delegate。然后在AFURLSessionManager中实现各个回调方法。最后把刚才创建的队列也设置上。

[self.session getTasksWithCompletionHandler:^(NSArray *dataTasks, NSArray *uploadTasks, NSArray *downloadTasks) {
}];
首先说说这个方法是干什么用的:这个方法用来异步的获取当前session的所有未完成的task。其实讲道理来说在初始化中调用这个方法应该里面一个task都不会有。我们打断点去看,也确实如此,里面的数组都是空的。但是想想也知道,AF大神不会把一段没用的代码放在这吧。所以我大胆猜测,可能是当我们重复去初始化session的时候(当然我们实际也不会这么做),会有新的session指向旧的有未完成task的session。为了排除这种不确定性因素,所以在初始化的时候把里面绑定的代理之类的东西都置为nil了。或许这就是防御性编程思想的一种体现吧。这段话是借鉴的别人的文章里的,正确性待验证。

四。一个简单的网络请求

调用AFHTTPSessionManager中的方法发起网络请求

    - (NSURLSessionDataTask *)GET:(NSString *)URLString
                   parameters:(id)parameters
                     progress:(void (^)(NSProgress * _Nonnull))downloadProgress
                      success:(void (^)(NSURLSessionDataTask * _Nonnull, id _Nullable))success
                      failure:(void (^)(NSURLSessionDataTask * _Nullable, NSError * _Nonnull))failure

    {

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

    [dataTask resume];    //启动网络请求
    return dataTask;
}

调用本类中的方法:dataTaskWithHTTPMethod:URLString:parameters:uploadProgress:downloadProgress:success:failure,此方法调用

- (NSURLSessionDataTask *)dataTaskWithHTTPMethod:(NSString *)method
                                      URLString:(NSString *)URLString
                                      parameters:(id)parameters
                                  uploadProgress:(nullable void (^)(NSProgress *uploadProgress)) uploadProgress
                                downloadProgress:(nullable void (^)(NSProgress *downloadProgress)) downloadProgress
                                        success:(void (^)(NSURLSessionDataTask *, id))success
                                        failure:(void (^)(NSURLSessionDataTask *, NSError *))failure
{

    NSError *serializationError = nil;
     //生成一个request,request包含method,url和参数等信息。request代表一个请求,NSURLSession对象用这个请求,生成一个task。然后执行这个task来完成一次网络请求。
    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
                          uploadProgress:uploadProgress
                        downloadProgress:downloadProgress
                      completionHandler:^(NSURLResponse * __unused response, id responseObject, NSError *error) {
        if (error) {
            if (failure) {
                failure(dataTask, error);
            }
        } else {
            if (success) {
                success(dataTask, responseObject);
            }
        }
    }];
    return dataTask;
}

然后调用基类AFURLSessionManager的方法

- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request
                              uploadProgress:(nullable void (^)(NSProgress *uploadProgress)) uploadProgressBlock
                            downloadProgress:(nullable void (^)(NSProgress *downloadProgress)) downloadProgressBlock
                            completionHandler:(nullable void (^)(NSURLResponse *response, id _Nullable responseObject,  NSError * _Nullable error))completionHandler {

    __block NSURLSessionDataTask *dataTask = nil;
    url_session_manager_create_task_safely(^{
        //初始化的时候就已经创建了session。见上文
        dataTask = [self.session dataTaskWithRequest:request];    
    });
    [self addDelegateForDataTask:dataTask uploadProgress:uploadProgressBlock downloadProgress:downloadProgressBlock completionHandler:completionHandler];

    return dataTask;
}

注释:
static方法 url_session_manager_create_task_safely(dispatch_block_t block)方法解决了NSURLSessionTask 的 taskIdentifier 在并发的情况下不唯一的bug,苹果在iOS8时解决了这个bug。 这个方法是同步的同步执行方法而已。

然后,session调用NSURLSession中的方法- (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request;生成NSURLSessionDataTask对象。

[dataTask resume]; 来启动网络请求

五。NSMutableURLRequest 对象的生成过程

在AFHTTPSessionManager中,会调用下面的方法,得到一个NSMutableURLRequest对象,这个方法就是创建和初始化此对象的。

- (NSMutableURLRequest *)requestWithMethod:(NSString *)method
                                URLString:(NSString *)URLString
                                parameters:(id)parameters
                                    error:(NSError *__autoreleasing *)error
{
    NSParameterAssert(method);
    NSParameterAssert(URLString);

    NSURL *url = [NSURL URLWithString:URLString];
    NSParameterAssert(url);
    NSMutableURLRequest *mutableRequest = [[NSMutableURLRequest alloc] initWithURL:url];
    mutableRequest.HTTPMethod = method;
    for (NSString *keyPath in AFHTTPRequestSerializerObservedKeyPaths()) {
        if ([self.mutableObservedChangedKeyPaths containsObject:keyPath]) {
            [mutableRequest setValue:[self valueForKeyPath:keyPath] forKey:keyPath];
        }
    }
    mutableRequest = [[self requestBySerializingRequest:mutableRequest withParameters:parameters error:error] mutableCopy];
return mutableRequest;
}

注释:

1.AFHTTPRequestSerializerObservedKeyPaths返回NSURLRequest相关的属性名称的数组

static NSArray * AFHTTPRequestSerializerObservedKeyPaths() {

    static NSArray *_AFHTTPRequestSerializerObservedKeyPaths = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{ _AFHTTPRequestSerializerObservedKeyPaths = @[NSStringFromSelector(@selector(allowsCellularAccess)), 
                    NSStringFromSelector(@selector(cachePolicy)), 
                    NSStringFromSelector(@selector(HTTPShouldHandleCookies)), 
                    NSStringFromSelector(@selector(HTTPShouldUsePipelining)), 
                    NSStringFromSelector(@selector(networkServiceType)), 
                    NSStringFromSelector(@selector(timeoutInterval))];
    });
    return _AFHTTPRequestSerializerObservedKeyPaths;
}

self.mutableObservedChangedKeyPaths某个request需要观察的属性集合,所以这段代码的意思是 在NSMutableURLRequest对象所有可以被观察的属性中,观察需要观察的属性。

2.调用AFHTTPRequestSerializer类中的方法。这个类遵循AFURLRequestSerialization协议(AFURLRequestSerialization是一个协议类型,不是基类)。这个方法设置并且返回了最终的NSMutableURLRequest类型的对象mutableRequest。

- (NSURLRequest *)requestBySerializingRequest:(NSURLRequest *)request
                              withParameters:(id)parameters
                                        error:(NSError *__autoreleasing *)error
{
    NSParameterAssert(request);
    NSMutableURLRequest *mutableRequest = [request mutableCopy];
    [self.HTTPRequestHeaders enumerateKeysAndObjectsUsingBlock:^(id field, id value, BOOL * __unused stop) {
        if (![request valueForHTTPHeaderField:field]) {
            [mutableRequest setValue:value forHTTPHeaderField:field];
        }
    }];
    NSString *query = nil;
    if (parameters) {
        if (self.queryStringSerialization) {
            NSError *serializationError;
            query = self.queryStringSerialization(request, parameters, &serializationError);
            if (serializationError) {
                if (error) {
                    *error = serializationError;
                }
                return nil;
            }
        } else {
            switch (self.queryStringSerializationStyle) {
                case AFHTTPRequestQueryStringDefaultStyle:
                    query = AFQueryStringFromParameters(parameters);
                    break;
            }
        }
    }
    if ([self.HTTPMethodsEncodingParametersInURI containsObject:[[request HTTPMethod] uppercaseString]]) {
        if (query && query.length > 0) {
            mutableRequest.URL = [NSURL URLWithString:[[mutableRequest.URL absoluteString] stringByAppendingFormat:mutableRequest.URL.query ? @"&%@" : @"?%@", query]];
        }
    } else {
        // #2864: an empty string is a valid x-www-form-urlencoded payload
        if (!query) {
            query = @"";
        }
        if (![mutableRequest valueForHTTPHeaderField:@"Content-Type"]) {
            [mutableRequest setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
        }

        [mutableRequest setHTTPBody:[query dataUsingEncoding:self.stringEncoding]];
    }
    return mutableRequest;
}

注释:

1.先看如下代码

[self.HTTPRequestHeaders enumerateKeysAndObjectsUsingBlock:^(id field, id value, BOOL * __unused stop) {
        if (![request valueForHTTPHeaderField:field]) {
            [mutableRequest setValue:value forHTTPHeaderField:field];
        }
    }];

这段儿代码的意思是,把需要放到request头里的东西,设置到request头里。self.HTTPRequestHeaders其实返回的是成员变量mutableHTTPRequestHeaders的不可变版本,类型是NSDictionary。mutableHTTPRequestHeaders里面放的是放到request的头中信息。比如:
"Accept-Language" = "en;q=1";
"User-Agent" = "AFNTest/1.0 (iPhone; iOS 11.2; Scale/3.00)";
等等。

未完待续

你可能感兴趣的:(AFNetworking源码简读)