iOS开发iOS DeveloperiOS 技术文档收录

NSURLCache 实现离线加载网页数据,离线缓存,节约流量,

2016-10-21  本文已影响318人  火柴大男人

最近做优化网页请求 研究了下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

上一篇下一篇

猜你喜欢

热点阅读