iOS网络——NSURLCache设置网络请求缓存

今天在看HTTP协议,看到了response头中的cache-control,于是就深入的研究了一下。发现了iOS中一个一直被我忽略的类——NSURLCache类。

NSURLCache

NSURLCache用于缓存网络请求,也就是NSURLRequest,然后根据我们设置的NSURLCache策略进行相应的缓存。

首先介绍一下各种策略

策略 意义

UseProtocolCachePolicy

 默认行为
 ReloadIgnoringLocalCacheData  不使用缓存
 ReloadIgnoringLocalAndRemoteCacheData*  我是认真地,不使用任何缓存
 ReturnCacheDataElseLoad  使用缓存(不管它是否过期),如果缓存中没有,那从网络加载吧
 ReturnCacheDataDontLoad  离线模式:使用缓存(不管它是否过期),但是从网络加载
 ReloadRevalidatingCacheData* 在使用前去服务器验证 

其中ReloadIgnoringLocalAndRemoteCacheData和ReloadRevalidatingCacheData两种是没有实现的,可以不看。

在创建对request使用cache的时候会让我们选择以上的某种策略进行,也就是

+ (instancetype)requestWithURL:(NSURL *)URL cachePolicy:(NSURLRequestCachePolicy)cachePolicy timeoutInterval:(NSTimeInterval)timeoutInterval;

该方法让我们设置策略和时间,然后request会根据策略和时间来进行相应的调度。

感受NSURLCache

这里使用默认的缓存策略ReturnCacheDataElseLoad缓存策略,

首先需要创建NSURLCache类

NSURLCache *URLCache = [[NSURLCache alloc] initWithMemoryCapacity:4 * 1024 * 1024 diskCapacity:0 diskPath:nil];
[NSURLCache setSharedURLCache:URLCache];

1.这里可以看到,创建参数我们制定了 4 * 1024 * 1024的内存(4MB) ,没有使用磁盘空间。

2.NSURLCache使用[NSURLCache sharedURLCache]创建默认的的缓存行为,默认为 4(MB) 内存和 20(MB)磁盘空间,这里我们使用自定义的,所以要setSharedCache。

然后创建request和connection进行请求

NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://172.16.25.44/test1.php"] cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:3];
connection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
[connection start];

实现NSURLConnectionDelegate协议

-(void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    NSLog(@"finish");
}

然后运行开一下请求,这里用的工具是Charles

可以看到只有一次请求,再看控制台输出

2015-08-04 09:29:55.297 requestCache[19405:6375355] finish
2015-08-04 09:29:55.301 requestCache[19405:6375355] finish
2015-08-04 09:29:55.310 requestCache[19405:6375355] finish
2015-08-04 09:29:55.451 requestCache[19405:6375355] finish
2015-08-04 09:29:55.618 requestCache[19405:6375355] finish
2015-08-04 09:29:55.784 requestCache[19405:6375355] finish
2015-08-04 09:29:55.984 requestCache[19405:6375355] finish
2015-08-04 09:29:56.120 requestCache[19405:6375355] finish

所以说多次的请求只会进行一次请求,因为在内存中NSURLCache为我们缓存了一份response,一旦有同样请求就会使用缓存。

缓存持久化

缓存如果设定本地磁盘就会为我们自动进行持久化,修改NSURLCache创建代码

    NSURLCache *URLCache = [[NSURLCache alloc] initWithMemoryCapacity:4 * 1024 * 1024 diskCapacity:20 * 1024 * 1024 diskPath:nil];
    [NSURLCache setSharedURLCache:URLCache];

设置了20MB的本地磁盘,然后运行程序,进行请求发现还是请求一次没有变化。但是在次运行程序进行请求就会发现,一次远程请求也不会进行了!

打开沙盒,发现在 Library/Caches/bundleId+项目名/下面有三个文件

这不就是sqlite么!原来NSURLCache帮我们用sqlite将请求存入了数据库,然后当有相同请求时就会调用缓存!

可以想到webView如果加载一个静态页面不用只用请求一次,并且在效果要更新的时候远程请求会有多爽!

默认策略

默认策略是 UseProtocolCachePolicy 从字面上来看是说,使用协议缓存策略,但是什么是协议缓存策略呢?

在HTTP协议的response头中,有一个字段是cache-control,由服务器来告诉客户端如何使用缓存。

下面是一个response头

可以看到cache-control指定的行为是public,max-age=5

这里先介绍一下各种指令

对应上表,可以看出了刚才响应头是要求缓存所有内容,缓存5秒失效,5秒后还要请求远程服务器。

对应PHP就是header("Cache-Control:public,max-age=5");

伪造响应

如果我们想让一些请求,有特定的响应,我们可以自己来制作响应

    NSLog(@"%@",[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject]);

    NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://172.16.25.44/test1.php"] cachePolicy:NSURLRequestReturnCacheDataElseLoad timeoutInterval:3];
    NSURLCache * cache = [NSURLCache sharedURLCache];

    NSData *contentData = [@"123" dataUsingEncoding:NSUTF8StringEncoding];

    NSURLResponse *response = [[NSURLResponse alloc] initWithURL:[NSURL URLWithString:@"http://172.16.25.44/test1.php"] MIMEType:@"text/html" expectedContentLength:1000 textEncodingName:@"UTF-8"];
    NSCachedURLResponse *cacheRespone = [[NSCachedURLResponse alloc] initWithResponse:response data:contentData];

    [cache storeCachedResponse:cacheRespone forRequest:request];

    connection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
    [connection start];

如上代码,创建了一个针对@"http://172.16.25.44/test1.php"请求的响应,并且让 cache 对该响应进行了存储。

实现

-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    NSString *dataString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    NSLog(@"%@",dataString);
}

输出结果如下

2015-08-04 09:48:58.825 requestCache[19503:6441561] 123
2015-08-04 09:48:58.826 requestCache[19503:6441561] finish
2015-08-04 09:48:58.983 requestCache[19503:6441561] 123
2015-08-04 09:48:58.984 requestCache[19503:6441561] finish
2015-08-04 09:48:59.167 requestCache[19503:6441561] 123
2015-08-04 09:48:59.167 requestCache[19503:6441561] finish
2015-08-04 09:48:59.334 requestCache[19503:6441561] 123
2015-08-04 09:48:59.335 requestCache[19503:6441561] finish

可以看到输出的是我们自定义的123,而不是服务器返回的1。

修改响应内容

修改响应内容需要我们实现NSURLConnectionDataDelegate 协议并实现

-(NSCachedURLResponse *)connection:(NSURLConnection *)connection willCacheResponse:(NSCachedURLResponse *)cachedResponse
{
    NSMutableData *mutableData = [[cachedResponse data] mutableCopy];

    //添加数据

    NSCachedURLResponse *response = [[NSCachedURLResponse alloc] initWithResponse:cachedResponse.response data:mutableData];
    return response;
}

应为 NSCachedURLResponse 的属性都是readonly的,所以我们想要添加内容就要创建一个可变副本增减内容。

时间: 2024-10-06 19:12:03

iOS网络——NSURLCache设置网络请求缓存的相关文章

linux网络基础设置 以及 软件安装

一. linux网络基础设置 网络设置基本可以分为:网卡设置.网关设置.DNS设置三部分 ifconfig命令 ifconfig命令被用于配置和显示Linux内核中网络接口的网络参数.用ifconfig命令配置的网卡信息,在网卡重启后机器重启后,配置就不存在.要想将上述的配置信息永远的存的电脑里,那就要修改网卡的配置文件了. 参数 add<地址>:设置网络设备IPv6的ip地址: del<地址>:删除网络设备IPv6的IP地址: down:关闭指定的网络设备: up:启动指定的网络

 iOS 网络请求缓存:NSURLCache详解

我读过一些开源项目的网络请求缓存的代码,基本上都是采用在本地存文件的方式进行缓存.如果你打算在你的项目中加入网络请求的缓存,可能你并不需要自己造一个轮子,了解一下 NSURLCache 就足够.本文为大家接收的就是ios开发中的NSURLCache相关使用,一起来看看吧. 缓存 首先, NSURLCache 提供的是内存以及磁盘的综合缓存机制.许多文章谈到,使用NSURLCache 之前需要在 AppDelegate 中缓存空间的设置: - (BOOL)application:(UIApplic

iOS开发中的网络请求

今天来说说关于iOS开发过程中的网络请求. 关于网络请求的重要性我想不用多说了吧.对于移动客户端来说,网络的重要性不言而喻.常见的网络请求有同步GET, 同步POST, 异步GET, 异步POST.今天来看一下四种网络请求的实现方式. 一.同步GET // 1.将网址初始化成一个OC字符串对象 NSString *urlStr = [NSString stringWithFormat:@"%@?query=%@&region=%@&output=json&ak=6E823

iOS NSURLConnection和异步网络请求

在日常应用中,我们往往使用AFNetworking等第三方库来实现网络请求部分.这篇文章会简要地介绍一下如何使用NSURLConnection来进行异步的网络请求. 我们先看一个小demo - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. NSString *urlStr = @"http://www.baidu.c

iOS 开发 把握AFNet网络请求完成的正确时机

前言 对于iOS开发中的网络请求模块,AFNet的使用应该是最熟悉不过了,但你是否把握了网络请求正确的完成时机?本篇文章涉及线程同步.线程依赖.线程组等专用名词的含义,若对上述名词认识模糊,可先进行查阅理解后阅读本文.如果你也纠结于文中所述问题,可进行阅读希望对你有所帮助.大神无视勿喷. 在真实开发中,我们通常会遇到如下问题: 一.某界面存在多个请求,希望所有请求均结束才进行某操作. 对于这一问题的解决方案很容易想到通过线程组进行实现.代码如下: 1 2 3 4 5 6 7 8 9 10 11

android 设置网络请求超时

UI界面更新必须在ui线程中 不能在ruanable线程中操作ui 可以发送消息利用handler来更新ui    private void load() {        LoadDate load = new LoadDate();        load.execute("http://h.hiphotos.baidu.com/image/w%3D310/sign=1bc9c0da38292df597c3aa148c305ce2/c83d70cf3bc79f3d77fbe1c5b8a1cd

iOS 自己封装的网络请求,json解析的类

基本上所有的APP都会涉及网络这块,不管是用AFNetWorking还是自己写的http请求,整个网络框架的搭建很重要. 楼主封装的网络请求类,包括自己写的http请求和AFNetWorking的请求,代码简单,主要是框架搭建.简单来说,就是一个请求类,一个解析类,还有若干数据类. 以下代码以公开的天气查询api为例: 1.网络请求类 我把常用的网络请求方法都封装好了,你只需要写自己的接口,传递apiName,params等参数就可以. #pragma mark ios请求方式 //ios自带的

iOS之NSURLCache、ASI设置缓存及使用步骤

1 一NSURLCache 2 缓存的使用步骤 3 // 获得全局的缓存对象 4 NSURLCache *cache = [NSURLCache sharedURLCache]; 5 6 // 设置缓存容量 7 cache.memoryCapacity = 1024 * 1024; 8 cache.diskCapacity = 20 * 1024 * 1024; 9 10 // 设置请求的缓存策略 11 request.cachePolicy = NSURLRequestReturnCacheD

移动开发在路上-- IOS移动开发系列 网络交互四(3)

接着上次的讲,这次我们讲 网络请求的封装  打开创建的项目,让我们一起来继续完成他, 上次我们说到GET请求地址的拼接: 我们接着昨天的继续完善: 下边我们要定义的是 block //定义block __block ASIFormDataRequest *request=[ASIFormDataRequest requestWithURL:url]; 请求超出时间 //设置请求超出时间 [request setTimeOutSeconds:60]; POST请求参数的处理 //处理POST请求方