美文网首页
AFNetworking学习笔记

AFNetworking学习笔记

作者: 希尔罗斯沃德_董 | 来源:发表于2022-01-13 17:06 被阅读0次

    为什么要使用AFNetworking

    AFNetworking线程安全模型是什么?如果管理网络请求和回调?

    HTTP请求封装AFHTTPSessionManager & AFURLSessionManager

    AFHTTPSessionManager对外接口,AFURLSessionManager负责主要逻辑。

    • 创建一个NSURLSessionTask(请求任务)和一个AFURLSessionManagerTaskDelegate(请求回调),传入的回调Block赋值给Delegate。
    • AFURLSessionManager创建时创建NSURLSession,同时指定一个队列,队列用的是NSOperationQueue,最大并发数是1,相当于串行队列,用于处理NSURLSession的代理方法。
    • 通过字典将请求和回调一一对应(key为task.taskIdentifier,value为delegate),其操作通过NSLock来保证字典线程安全。
    • 当请求回调时,通过task.taskIdentifier从字典中取出对应的Delegate,处理回调。此时的代理方法时在NSURLSession的NSOperationQueue里面执行的。然后Delegate将完成的Block加入到回调队列里面,并将队列任务进行分组。
    • 保证了请求和回调一一对应,同时回调都在指定的队列执行。Group可以让我们在需要任务分组时使用。
    请求和回调的序列化
    • AFURLRequestSerialization 请求序列化,主要是将参数(字典)根据请求类型写入url或者body中。
    • AFURLResponseSerialization 实际上就是网络数据解析,支持包括json解析和xml解析。还支持图片数据转换成UIImage.
    Hearder配置

    使用dispatch_sync(读)+dispatch_barrier_async(写)+并行队列的组合来保障Hearder的线程安全。

    通过KVO管理一些通用属性的设置
    - (void)setAllowsCellularAccess:(BOOL)allowsCellularAccess {
        [self willChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
        _allowsCellularAccess = allowsCellularAccess;
        [self didChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
    }
    
    - (void)setCachePolicy:(NSURLRequestCachePolicy)cachePolicy {
        [self willChangeValueForKey:NSStringFromSelector(@selector(cachePolicy))];
        _cachePolicy = cachePolicy;
        [self didChangeValueForKey:NSStringFromSelector(@selector(cachePolicy))];
    }
    
    - (void)setHTTPShouldHandleCookies:(BOOL)HTTPShouldHandleCookies {
        [self willChangeValueForKey:NSStringFromSelector(@selector(HTTPShouldHandleCookies))];
        _HTTPShouldHandleCookies = HTTPShouldHandleCookies;
        [self didChangeValueForKey:NSStringFromSelector(@selector(HTTPShouldHandleCookies))];
    }
    
    - (void)setHTTPShouldUsePipelining:(BOOL)HTTPShouldUsePipelining {
        [self willChangeValueForKey:NSStringFromSelector(@selector(HTTPShouldUsePipelining))];
        _HTTPShouldUsePipelining = HTTPShouldUsePipelining;
        [self didChangeValueForKey:NSStringFromSelector(@selector(HTTPShouldUsePipelining))];
    }
    
    - (void)setNetworkServiceType:(NSURLRequestNetworkServiceType)networkServiceType {
        [self willChangeValueForKey:NSStringFromSelector(@selector(networkServiceType))];
        _networkServiceType = networkServiceType;
        [self didChangeValueForKey:NSStringFromSelector(@selector(networkServiceType))];
    }
    
    - (void)setTimeoutInterval:(NSTimeInterval)timeoutInterval {
        [self willChangeValueForKey:NSStringFromSelector(@selector(timeoutInterval))];
        _timeoutInterval = timeoutInterval;
        [self didChangeValueForKey:NSStringFromSelector(@selector(timeoutInterval))];
    }
    // 手动通知 https://www.jianshu.com/p/c2cf7cde8397
    + (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key {
        if ([AFHTTPRequestSerializerObservedKeyPaths() containsObject:key]) {
            return NO;
        }
    
        return [super automaticallyNotifiesObserversForKey:key];
    }
    
    - (void)observeValueForKeyPath:(NSString *)keyPath
                          ofObject:(__unused id)object
                            change:(NSDictionary *)change
                           context:(void *)context
    {
        if (context == AFHTTPRequestSerializerObserverContext) {
            if ([change[NSKeyValueChangeNewKey] isEqual:[NSNull null]]) {
                [self.mutableObservedChangedKeyPaths removeObject:keyPath];
            } else {
                [self.mutableObservedChangedKeyPaths addObject:keyPath];
            }
        }
    }
    

    备注:KVO观察分为自动观察和手动观察。被观察对象通过automaticallyNotifiesObserversForKey方法判断是否是自动观察。默认情况下返回YES,即是自动观察,比如我们前面的代码。那什么情况下是手动观察呢?为了实现手动观察我们得让被观察对象重写automaticallyNotifiesObserversForKey:,这里可以指定某个Key为手动观察。但是光实现这个方法还不够,我们还要手动调用willChangeValueForKey和didChangeValueForKey方法以通知观察者属性的变化。

    网络状态监测
    • AFNetworkReachabilityManager 检查网络状态
      SCNetworkReachabilityRef(address or domainName + SCNetworkReachabilityContext + SCNetworkReachabilitySetCallback)+ Runloop 监测网络的可达性,然后通过SCNetworkReachabilitySetCallback设置的block进行回调。
    安全策略
    • AFSecurityPolicy处理https相关的公钥和验证逻辑。目前由于苹果ATS的开启,基本HTTPS已经成为标配。
    • 什么是中间人攻击?
      大体就是黑客通过截获服务器返回的证书,并伪造成自己的证书,通常我们使用的Charles/Fiddler等工具实际上就可以看成中间人攻击。
    • AFSecurityPolicy如何避免中间人攻击?
      解决方案其实也很简单,就是SSL Pinning。AFSecurityPolicy的AFSSLPinningMode就是相关设置项。
      SSL Pinning的原理就是需要将服务器的公钥打包到客户端中,tls验证时,会将服务器的证书和本地的证书做一个对比,一致的话才允许验证通过。
      typedef NS_ENUM(NSUInteger, AFSSLPinningMode) {
      AFSSLPinningModeNone,
      AFSSLPinningModePublicKey, // 只验证证书中的公钥
      AFSSLPinningModeCertificate, // 验证证书所有字段,包括有效期之内
      };
      由于数字证书存在有效期,内置到客户端后就存在失效后导致验证失败的问题,所以可以考虑设置为AFSSLPinningModePublicKey的模式,这样的话,只要保证证书续期后,证书中的公钥不变,就能够通过验证了。
    UIKit模块,主要实现网络图片的加载和缓存
    • 通过UIKit相关类的Category封装接口,方便调用。

    • 下载任务和回调管理
      图片下载有专门的下载器AFImageDownloader来负责,AFImageDownloaderMergedTask是下载任务,一对多的形式对应多个回调,这样多个url相同的图片可以复用一个请求,优化流量。利用同步串行管理下载队列,通过对大并发数量和活跃数量来管理队列任务的执行。通过异步+串行队列处理图片下载完成的回调,将图片数据转换成UIImage然后在回到主线程,调用task对应的所有回调。

    • 内存缓存实现AFAutoPurgingImageCache

      • 缓存实现 使用NSMutableDictionary。
      • 线程安全模型dispatch_sync + dispatch_barrier_async + DISPATCH_QUEUE_CONCURRENT,实现多读单写,提高读取图片的效率。
    • 磁盘缓存实现NSURLCache
      NSURLCache是系统网络请求缓存,NSURLCache缓存的实际上是按照NSURLRequest 和Response的映射关系缓存。包括内存缓存和磁盘缓存。缓存策略默认是用的NSURLRequest的缓存策略。我们可以通过自己设置缓存大小和缓存路径。

    • 提供取消先下载任务的接口

    相关文章

      网友评论

          本文标题:AFNetworking学习笔记

          本文链接:https://www.haomeiwen.com/subject/egsacrtx.html