NSURLCache 实现离线加载网页数据,离线缓存,节约流量,
最近做优化网页请求 研究了下NSURLCache 实现离线加载网页数据
代码: KLURLCache
博客:@火柴大男人
网上找了很多资料,写的好的是一个外国人写的但也是2012年写的..很多东西已经过时 代码风格也是MRC.
国内有些人翻译过来,自己写成中文版,看了一下也没有DEMO代码. 查了其它的一些文章试了下,大多只是在讲理论.都 没有代
码DMEO.感觉研究起来费时间. 后来找一份实现代码.也有相关 博客说明.代码是MRC 有些方法也过期了..本人就是参考那份
代码实现,并做了一些改进.
写文章前一直找当时下载代码的地方 和博客地址 希望 注明一下引用一下.以示对原作者的尊重..但由于当时没有标记位置.现在也无法找到当时出处.所以在此谢过原作者的开源精神!!
第一步:
创建一个类继承 NSURLCache
@interface KLURLCache : NSURLCache
重写这个方法,,该方法是设置 内存 与 磁盘缓存 空间大小的
- (instancetype)initWithMemoryCapacity:(NSUInteger)memoryCapacity diskCapacity:(NSUInteger)diskCapacity diskPath:(NSString *)path{
if (self = [super initWithMemoryCapacity:memoryCapacity diskCapacity:diskCapacity diskPath:path]) {
if (!path) {
_diskPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
_responseInfoDict = [NSMutableDictionary dictionaryWithCapacity:0];
}else{
_diskPath = path;
}
}
return self;
}
第二步:
关键的一步
重写 这个方法 每个请求都 会经过这个方法.做缓存就是拦截这个方法 只要是GET请求才能缓存的
- (NSCachedURLResponse *)cachedResponseForRequest:(NSURLRequest *)request{
NSLog(@"%@",request.HTTPMethod);
if ([request.HTTPMethod compare:@"GET"] != NSOrderedSame) {
return [super cachedResponseForRequest:request];
}else{
return [self KL_CacheResponseDataFromRequest:request];
}
第三步:
缓存是如果做的 之前 请求都是通过[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse *response, NSData *data,NSError *error)进行的
IOS 9.0 后网络请求改为NSURLSession 进行管理.而不是之前的 NSURLConnection 原代码是NSURLConnection 代码也是MRC 这个地方自己修改了;
我们拿到 全局 NSURLSession *session = [NSURLSession sharedSession];
session 请求来的数据我们写入磁盘.
关于那个字典是什么,我们是缓存请求数据类型时间 键值对的.. 到时取缓存的时侯时需要用到的
NSURLSession *session = [NSURLSession sharedSession];
NSURLSessionTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
if (response && data) {
[self.responseInfoDict removeObjectForKey:fileName];
}
if (error) {
cachedResponse = nil;
}
NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:[NSString stringWithFormat:@"%f", [date timeIntervalSince1970]], @"time",response.MIMEType,@"MIMEType",response.textEncodingName,@"textEncodingName" ,nil];
[dict writeToFile:otherInfoPath atomically:YES];
[data writeToFile:filePath atomically:YES];
cachedResponse = [[NSCachedURLResponse alloc]initWithResponse:response data:data];
}];
一个GET请求 类似于 键值对一样..因为一个网页不是简单的请求一个 www.baidu.com 这么简单..请求这个后里面还会产生很多小请求加载数据的..每次请求都 会有一个request 我们所做的就是 根据当前request里面的URL 做一对一的缓存.
比如打开百度时 www.baidu.com 这个URL地址已经缓存了,那么当request的URL地址是这个的时侯我们就不会向风络再发请求.而是从本地沙盒中拿相应的数据返回给到它..但百度www.baidu.com后里面可能还www.baidu.com/1323.jpg 等等.而且每次刷新在www.baidu.com里面会有新的数据.URL请求,这个时侯可能就会出现第二次打开www.baidu.com时 些是从网络请求的.有些是从本地加载的.如下:
2016-10-21 16:57:09.928 DEMO[1924:698294] 请求的数据是---http://www.baidu.com/
2016-10-21 16:57:09.928 DEMO[1924:698294] data from Cache
......地址是----http://m.baidu.com/static/ecom/js/wise/home/ecomAds_5f3f8bc.js
2016-10-21 16:57:09.935 DEMO[1924:698304] GET
2016-10-21 16:57:09.935 DEMO[1924:698304] 请求的数据是---http://www.baidu.com/
2016-10-21 16:57:09.936 DEMO[1924:698304] data from Cache
......地址是----http://m.baidu.com/static/search/baiduapp_icon.png
2016-10-21 16:57:09.937 DEMO[1924:698351] GET
2016-10-21 16:57:09.937 DEMO[1924:698351] 请求的数据是---http://www.baidu.com/
2016-10-21 16:57:09.938 DEMO[1924:698351] data from request
地址是------http://m.baidu.com/tc?tcreq4log=1&ssid=0&from=844b&pu=sz%401320_2001%2Cta%40iphone_1_9.3_3_601&qid=15764434222633284224-2&ct=21&cst=1&ref=index_iphone&lid=15764434222633284224-2&w=0_0_&sid=110316_104496_100186_102628_104381_100100_108470_110005_107800_110394_109974_110420_110498_110011_110298_110251_109588_108013_107320_110357_110031_110270_110400&sOS=1&banner=wiseindex&r=1477040229894
2016-10-21 16:57:09.939 DEMO[1924:698351] GET