iOS 网络缓存


常见的网络数据缓存方式

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网络缓存扫盲
  • 网络缓存优化

你可能感兴趣的:(iOS 网络缓存)