AFHTTPRequestOperationManager
TL;DR
Manager 做的就是一个管理统筹者的工作。
利用 requestSerializer 来创建 request,然后基于 request 创建 operation。
然后将 operation 添加入队列,就完成了所有的使命。
- Clang Diagnostic
- NSSecureCoding
- NSCopying
概述
基于 NSURLConnection 系统环环相扣的设计,AFHTTPRequestOperationManager 也吸取了他们的设计思路:以 Operation 为核心操作单位的任务调度系统。
每一次网络请求都是基于一个 AFHTTPRequestOperation 实例
创建一次请求的流程如下:
Parameters
-> 由 <AFURLRequestSerialization>
-> NSMutableURLRequest
-> AFHTTPRequestOperation
-> Manager.operationQueue.addOperation
在 GET 等方法中,都是直接利用 self.requestSerializer 来构建 request,这个我们后面说到 Serializer 的时候再详细说。
Clang Diagnostic警告
#pragma clang diagnostic push // 保存编译器状态
#pragma clang diagnostic ignored "-Wgnu" // 忽略一个警告
// code here
#pragma clang diagnostic pop // 恢复编译器状态
AFURLConnectionOperation
使用 Operation 之前,AFN 先指定了:
operation.responseSerializer = self.responseSerializer;
operation.shouldUseCredentialStorage = self.shouldUseCredentialStorage;
operation.credential = self.credential;
operation.securityPolicy = self.securityPolicy;
[operation setCompletionBlockWithSuccess:success failure:failure];
operation.completionQueue = self.completionQueue;
operation.completionGroup = self.completionGroup;
AFURLConnectionOperation 并不是一个被直接使用的 Class,而是作为一个执行网络请求的基类存在,在 AFN 的实际使用中主要是他的子类AFHTTPRequestOperation
在进行更多细节方面的处理。
他的核心就是 NSOperation 的子类,并且实现了 NSURLConnection 的 delegate。然后通过传入的 block 给上层返回数据。
NSSecureCoding
要点:
+ (BOOL)supportsSecureCoding
- (id)initWithCoder:
- (void)encodeWithCoder:
- 安全
NSSecureCoding 是一个相对比较陌生的协议,他是 NSCoding 的补充协议,除了可以用来 archive 一个对象以外,还有一个方法 + (BOOL)supportsSecureCoding
来声明了它可以处理本身实例的编码解码方式,以防止替换攻击。
NSSecureCoding 对于 NSCoding,核心就是 Secure 在解码阶段指定了内容的类型。
这样做的好处也很明显:因为你不知道你拿来解码的内容是不是会导致崩溃或攻击。
我们来对比一下 NSCoding 和 NSSecureCoding 的实现区别:
NSCoding | NSSecureCoding | |
---|---|---|
- initWithCoder: | self.title = [decoder decodeObjectForKey:@"title"]; |
id obj = [decoder decodeObjectOfClass:[MyClass class] forKey:@"title"]; |
- encodeWithCoder: | [encoder encodeObject:self.title forKey:@"title"]; |
同左 |
+ supportsSecureCoding | 无 | 必须返回 true / YES |
如果 NSCoding 需要避免上述问题,就需要做一次额外的检查:
id obj = [decoder decodeObjectForKey:@"title"];
if (![obj isKindOfClass:[MyClass class]]) {
// fail
}
NSCopying
要点:
- (id)copyWithZone:
实现 copyWithZone
就可以在代码中使用 [id copy] 方法来对对象进行复制,具体是怎样的执行细节,需要你在该方法中自己实现。
Question
- 为什么会有这样的写法?
// .h
@property (readonly, nonatomic, strong, nullable) NSURL *baseURL;
// .m
@property (readwrite, nonatomic, strong) NSURL *baseURL;
我觉得使用 _baseURL 可以实现同样的效果,这里用 extension 覆盖掉 h 文件中的声明,应该是为了在 m 文件中代码风格保持一致