GET 缓存导致的问题
在iOS 客户端通过AFNetworking 发起 GET 方式的HTTP请求时,导致数据始终不是最新的,即使断开网络连接,依旧能获取到response,原因就是其对GET 方式进行了缓存。
根据HTTP规范,GET用于信息获取,而且应该是安全的和幂等的 。
1.所谓安全的意味着该操作用于获取信息而非修改信息。换句话说,GET请求一般不应产生副作用。就是说,它仅仅是获取资源信息,就像数据库查询一样,不会修改,增加数据,不会影响资源的状态。
* 注意:这里安全的含义仅仅是指是非修改信息。
2.幂等的意味着对同一URL的多个请求应该返回同样的结果.
由于get是幂等的, 关于get和post的区别中有一条为:
从缓存的角度,GET 请求会被浏览器主动缓存下来,留下历史记录,而 POST不会。
HTTP缓存
HTTP缓存的基本目的就是使应用执行的更快,更易扩展,但是HTTP缓存通常只适用于idempotent request(可以理解为查询请求,也就是不更新服务端数据的请求),这也就导致了在HTTP的世界里,一般都是对GET请求做缓存,POST请求很少有缓存。
GET多用来直接获取数据,不修改数据,主要目的就是database的search语句的感觉。用缓存(有个代理服务器的概念)的目的就是查db的速度变快。
POST则是发送数据到服务器端去存储。类似db里的update delete和insert语句的感觉。更新db的意思。数据必须放在数据库,所以一般都得去访问服务器端,而极少需要缓存。
把时间调到一个过去的时间, get请求就会使用cache中缓存的东西, 尝试一个通用的解法, 不让使用缓存,
1.客户端在header中添加@{@"If-Modified-Since": @"0"}
2.客户端在header中添加@{@"Cache-Control": @"no-cache”},
3.客户端在GET请求的url中添加参数@{@"timestamp": @([[NSDate date] timeIntervalSince1970]} , (推荐)
4.iOS 可以通过使用[NSURLSessionConfiguration ephemeralSessionConfiguration]忽略缓存
defaultSessionConfiguration: 默认 session 配置,类似 NSURLConnection 的标准配置,使用硬盘来存储缓存数据。
ephemeralSessionConfiguration: 临时session配置,与默认配置相比,这个配置不会将缓存、cookie等存在本地,只会存在内存里,所以当程序退出时,所有的数据都会消失
5.修改URLCaChe
可以清除所有的本地缓存
[[NSURLCache sharedURLCache] removeAllCachedResponses];
也可以把cache的的值设置为0;系统默认在内存上分配约512KB
的空间,在磁盘上分配约10M
的空间。 // 确认有效
[[NSURLCache sharedURLCache] setMemoryCapacity:0];
[[NSURLCache sharedURLCache] setDiskCapacity:0];
6.发起的request请求可以设置为 reloadIgnoringLocalAndRemoteCacheData
request.cachePolicy = NSURLRequestReloadIgnoringLocalCacheData; // 拼接request 是在AFN内部做的,影响较大
默认值为NSURLRequestUseProtocolCachePolicy , 按照协议缓存, 所以服务端修改Cache-Control也是生效的
需要服务端处理:
1.服务端处理,在response的 header 中添加@{@"Cache-Control": @"no-cache, must-reva lidate"};
2.用POSET替换GET请求,所有出问题的地方都是因为不符合get请求的语义, get请求不符合幂等的概念(推荐)
参考链接: