前言
Mattt Thompson 在NSHipster中介绍NSURLCache时说,无数开发者尝试自己做一个简陋而脆弱的系统来实现网络缓存的功能,殊不知 NSURLCache
只要两行代码就能搞定且好上100倍。甚至更多开发者根本不知道网络缓存的好处,也从未尝试过,导致他们的应用向服务器作了无数不必要的网络请求。在看完UIImageView+AFNetworking
中的实现后,不禁万分惊喜。
一、 NSURLCache
介绍
NSURLCache
提供对URL请求的响应的磁盘缓存和内存缓存,作为 URL Loading System
的一部分,任何通过 NSURLSession
加载的请求都将被 NSURLCache
处理。网络缓存减少了需要向服务器发送请求的次数,同时也提升了离线或在低速网络中使用应用的体验。当一个请求完成来自服务器的响应,这个响应将在本地和内存中保存。下一次同一个请求再发起时,本地保存的响应就会马上返回,不需要连接服务器。
二、NSURLCache
缓存策略
关于缓存的使用,我们先来看看系统提供的缓存策略:NSURLRequestCachePolicy
在发起网络请求时,我们有多种方式去设置请求的缓存策略
// 初始化NSURLRequest配置
+ (instancetype)requestWithURL:(NSURL *)URL cachePolicy:(NSURLRequestCachePolicy)cachePolicy timeoutInterval:(NSTimeInterval)timeoutInterval;
// 初始化NSRURLSession时NSURLSessionConfiguration配置
@property NSURLRequestCachePolicy requestCachePolicy;
缓存策略枚举:
typedef NS_ENUM(NSUInteger, NSURLRequestCachePolicy)
{
// 该策略为协议特定(默认的缓存策略)
NSURLRequestUseProtocolCachePolicy = 0,
// 忽略本地缓存
NSURLRequestReloadIgnoringLocalCacheData = 1,
// 未实现
NSURLRequestReloadIgnoringLocalAndRemoteCacheData = 4, // Unimplemented
// 忽略本地缓存
NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData,
// 使用本地缓存,本地无缓存才使用网络加载
NSURLRequestReturnCacheDataElseLoad = 2,
// 使用本地缓存,永远不使用网络加载,相当于离线模式
NSURLRequestReturnCacheDataDontLoad = 3,
// 未实现
NSURLRequestReloadRevalidatingCacheData = 5, // Unimplemented
};
目前看来,要想使用缓存,我们在缓存策略中只有2种选择:
// 该策略为协议特定(默认的缓存策略)
NSURLRequestUseProtocolCachePolicy = 0,
// 使用本地缓存,本地无缓存才使用网络加载
NSURLRequestReturnCacheDataElseLoad = 2,
三、NSURLCache
默认缓存策略的使用
// 该策略为协议特定(默认的缓存策略)
NSURLRequestUseProtocolCachePolicy = 0,
NSURLRequestUseProtocolCachePolicy
官方文档给出的解释是缓存使用HTTP协议语义,此策略需要配合Web服务器进行缓存管理,在HTTP协议的NSHTTPURLResponse头中的cache-control字段,由服务器来告诉客户端如何使用缓存。Last-Modified/If-Modified-Since
和 Etag/If-None-Match
控制是否需要更新缓存。
"Cache-Control" ":" cache-directive
作为请求首部时,cache-directive 的可选值有:
作为响应首部时,cache-directive 的可选值有:
Last-Modified/If-Modified-Since
Last-Modified
是由服务器返回响应头,标识资源的最后修改时间.
If-Modified-Since
则由客户端发送,标识客户端所记录的,资源的最后修改时间。服务器接收到带有该请求头的请求时,会使用该时间与资源的最后修改时间进行对比,如果发现资源未被修改过,则直接返回HTTP 304而不返回包体,告诉客户端直接使用本地的缓存。否则响应完整的消息内容。
Etag/If-None-Match
Etag
由服务器发送,告之当资源在服务器上的一个唯一标识符。
客户端请求时,如果发现资源过期(使用Cache-Control的max-age
),发现资源具有Etag
声明,这时请求服务器时则带上If-None-Match
头,服务器收到后则与资源的标识进行对比,决定返回200或者304。
max-age
和Expire
都代表缓存有效期,当两个字段都存在时,max-age
会覆盖Expire
。
四、NSURLCache NSURLRequestReturnCacheDataElseLoad
使用
// 使用本地缓存,本地无缓存才使用网络加载
NSURLRequestReturnCacheDataElseLoad = 2,
使用这种策略缓存只需要将 requestCachePolicy配置为NSURLRequestReturnCacheDataElseLoad就行了,我们不需要做其他操作,但是这种策略有一个缺点,当缓存存在的情况下,如果服务器资源发送变化,缓存不会更新,客户端一直源用之前的缓存,这种方案只适用于资源不会发生变化的情况,比如图片资源等。
五、NSURLCache
使用的注意点
1.在APP启动时我们应该设置一个共享缓存磁盘大小来控制缓存可以占用的空间
- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
NSURLCache *URLCache = [[NSURLCache alloc] initWithMemoryCapacity:memorySize * 1024 * 1024
diskCapacity:diskSize * 1024 * 1024
diskPath:nil];
[NSURLCache setSharedURLCache:URLCache];
}
2.同时在设置缓存空间大小后需要注意单次缓存不能超过给定的总的磁盘缓存大小的%5.超过%5将不会被缓存
The response size is small enough to reasonably fit within the cache. (For example, if you provide a disk cache, the response must be no larger than about 5% of the disk cache size.)
3.NSURLSessionDataDelegate
中可以修改缓存和换回nil不使用缓存
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
willCacheResponse:(NSCachedURLResponse *)proposedResponse
completionHandler:(void (^)(NSCachedURLResponse * _Nullable cachedResponse))completionHandler{
}