常见的网络数据缓存方式
GET网络请求缓存
概述
首先要知道,POST请求不能被缓存,只有 GET 请求能被缓存。缓存的思路就是将查询的参数组成的值作为 key ,对应结果作为value。从这个意义上说,一个文件的资源链接,也叫 GET 请求
该怎么做?
前提
针对的是get请求
针对的是get请求
针对的是get请求
NSURLCache : NSURLCache 为您的应用的 URL 请求提供了内存中以及磁盘上的综合缓存机制。 作为基础类库 URL 加载系统 的一部分,任何通过 NSURLConnection 加载的请求都将被 NSURLCache 处理。个默认缓存在内存,并且可以通过一些配置操作可以持久缓存到磁盘的类。
那么知道了这些之后开始我们的步骤
设置NSURLCache的大小时,大多使用下面的代码
- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
NSURLCache *urlCache = [[NSURLCache alloc] initWithMemoryCapacity:4 * 1024 * 1024 diskCapacity:20 * 1024 * 1024 diskPath:nil];
[NSURLCache setSharedURLCache:urlCache];
}
设置一个缓存策略
无论你是用的是NSURLRequest还是NSMutableURLRequest,你都需要去设置一下它们的cachePolicy属性
typedef NS_ENUM(NSUInteger, NSURLRequestCachePolicy)
{
NSURLRequestUseProtocolCachePolicy = 0,
NSURLRequestReloadIgnoringLocalCacheData = 1,
NSURLRequestReloadIgnoringLocalAndRemoteCacheData = 4, // Unimplemented
NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData,
NSURLRequestReturnCacheDataElseLoad = 2,
NSURLRequestReturnCacheDataDontLoad = 3,
NSURLRequestReloadRevalidatingCacheData = 5, // Unimplemented
}
这些缓存都代表什么意思呢?
NSURLRequestUseProtocolCachePolicy: 对特定的 URL 请求使用网络协议中实现的缓存逻辑。这是默认的策略。
NSURLRequestReloadIgnoringLocalCacheData:数据需要从原始地址加载。不使用现有缓存。
NSURLRequestReloadIgnoringLocalAndRemoteCacheData:不仅忽略本地缓存,同时也忽略代理服务器或其他中间介质目前已有的、协议允许的缓存。
NSURLRequestReturnCacheDataElseLoad:无论缓存是否过期,先使用本地缓存数据。如果缓存中没有请求所对应的数据,那么从原始地址加载数据。
NSURLRequestReturnCacheDataDontLoad:无论缓存是否过期,先使用本地缓存数据。如果缓存中没有请求所对应的数据,那么放弃从原始地址加载数据,请求视为失败(即:“离线”模式)。
NSURLRequestReloadRevalidatingCacheData:从原始地址确认缓存数据的合法性后,缓存数据就可以使用,否则从原始地址加载
知道了它们的意思之后我把工程中加入了下面的代码
if (![Global shareInstance].isNetReachable) {
request.cachePolicy = NSURLRequestReturnCacheDataDontLoad;
}else{
request.cachePolicy = NSURLRequestReturnCacheDataElseLoad;
}
这样就做到了当网络不通时使用本地缓存。当有网络是则先使用本地缓存,如果找不到本地缓存就去加载数据。
但有些场景下我们的缓存是存在时效性的。比如商品详情不同的时间返回的价格因为有活动价已经不同了。那么此时这种办法就明显不适用了。
控制缓存的有效性
借助ETag或者Las-Modified判断文件是否有效
ETag
HTTP协议规格说明定义ETag为“被请求变量的实体值”。另一种说法是,ETag是一个可以与Web资源关联的记号(token)。Web资源可以是一个web页面、json或xml数据、文件等。Etag有点类似于文件hash或者说是信息摘要。
在浏览器默认的行为中,当进行一次URL请求,服务端会返回'Etag'响应头,下次浏览器请求相同的URL时,浏览器会自动将它设置为请求头'If-None-Match'的值。服务器收到这个请求之后,就开始做信息校验工作将自己本次产生的Etag与请求传递过来的'If-None-Match'对比,如果相同,则返回HTTP状态码304,并且response数据体中没有数据。
具体的使用看下面代码例子
- (void)getData:(GetDataCompletion)completion {
NSURL *url = [NSURL URLWithString:kETagImageURL];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url cachePolicy:NSURLRequestReloadIgnoringCacheData timeoutInterval:15.0];
// 发送 etag
if (self.etag.length > 0) {
[request setValue:self.etag forHTTPHeaderField:@"If-None-Match"];
}
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
// NSLog(@"%@ %tu", response, data.length);
// 类型转换(如果将父类设置给子类,需要强制转换)
NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response;
NSLog(@"statusCode == %@", @(httpResponse.statusCode));
// 判断响应的状态码是否是 304 Not Modified (更多状态码含义解释: https://github.com/ChenYilong/iOSDevelopmentTips)
if (httpResponse.statusCode == 304) {
NSLog(@"加载本地缓存图片");
// 如果是,使用本地缓存
// 根据请求获取到`被缓存的响应`!
NSCachedURLResponse *cacheResponse = [[NSURLCache sharedURLCache] cachedResponseForRequest:request];
// 拿到缓存的数据
data = cacheResponse.data;
}
// 获取并且纪录 etag,区分大小写
self.etag = httpResponse.allHeaderFields[@"Etag"];
NSLog(@"etag值%@", self.etag);
!completion ?: completion(data);
}];
}
Last-Modified
Last-Modified值在服务器处理阶段代表着文件的上次修改时间,在处理结束后作为一个响应头放到response中。如果在请求中添加了'If-Modified-Since'头,并将这个值设置为上次请求时得到的响应头'Last-Modified'的值,那么这次请求服务器中逻辑的伪代码
if ETag != 请求头中的'If-Non-Match' || 查询到的'Last-Modified'(上次修改的时间) != 请求头中的'If-Modified-Since'
返回的response状态码200 和 数据
else
返回的reponse状态码304
目前这些也是个人通过学习网上文章的理解,待之后实践后再来补充详细理解。
关于ETag和Last-Modified的详细说明可以看下面两边详细讲解
- iOS网络缓存扫盲
- 网络缓存优化