最近做了UIWebView的缓存,用了两种方式去实现:第一种使用的是重写UIWebview的URlCache方法,在urlCache方法中请求网页数据,保存网页数据至本地磁盘。
使用RNCachingURLProtocol协议:优点:不会像第一种方式会在每次使用缓存的时候产生两个相同的请求,缺点:会缓存代码中所有的URL请求。对于不需要进行缓存的URL在请求头中加上X-RNCache,协议中的代码中写有注释,
+ (BOOL)canInitWithRequest:(NSURLRequest *)request
{
// only handle http requests we haven't marked with our header.
if ([[self supportedSchemes] containsObject:[[request URL] scheme]] &&
([request valueForHTTPHeaderField:RNCachingURLHeader] == nil))
{
return YES;
}
return NO;
}
使用的是AF库加请求头的方法:
//包含该头时,RNCachingURLProtocol不使用缓存
[manager.requestSerializer setValue:@"YES" forHTTPHeaderField:@"X-RNCache"];
由于使用的是协议,所以在控制什么时候去执行缓存,什么时候不需要缓存不能直接在使用的时候去调用方法,使用的是用NSUserDefault存储值的方法去控制是否需要缓存,在使用webview
的缓存需要注意的几点:对于不需要缓存的请求需要手动在进行缓存的方法中过滤掉,要不在下次执行的时候会主动检查缓存不会再向服务器发起请求,在这个库中主要用到的几个方法:
//是否需要使用缓存
- (BOOL) useCache
{
// isReload:0不重新加载 1重新加载
int isReload =0;
if(![ToolisNewWorkConnected])
{
returnYES;
}
if([[NSUserDefaultsstandardUserDefaults] valueForKey:ISRELOADWEBVIEW])
{
isReload = [[[NSUserDefaultsstandardUserDefaults] valueForKey:ISRELOADWEBVIEW]intValue];
}
//判断本地是否有缓存,有:取缓存否:从网络加载
NSString *cachePath = [selfcachePathForRequest:[selfrequest]];
NSFileManager *manager = [NSFileManagerdefaultManager];
if(isReload)
{
returnNO;
}
elseif([manager fileExistsAtPath:cachePath])
{
returnYES;
}
else
{
returnNO;
}
}
//保存浏览的数据
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
[[selfclient] URLProtocolDidFinishLoading:self];
NSString *cachePath = [selfcachePathForRequest:[selfrequest]];
RNCachedData *cache = [RNCachedDatanew];
[cache setResponse:[selfresponse]];
[cache setData:[selfdata]];
NSString *requestUrl =self.request.URL.absoluteString;
BOOL isNotNeedCache = [[NSUserDefaultsstandardUserDefaults] boolForKey:ISNOTNEEDCACHE];
//将不需要缓存的方法过滤掉
if([requestUrlrangeOfString:@"getarticlecount"].length ==0 && [requestUrl rangeOfString:@"isreadbyuser"].length ==0 &&[requestUrl rangeOfString:@"item_status"].length ==0 &&[requestUrl rangeOfString:@"article_web_status"].length ==0&&[requestUrl rangeOfString:@"ad_web_status"].length ==0 && !isNotNeedCache)
{
[NSKeyedArchiverarchiveRootObject:cache toFile:cachePath];
}
[selfsetConnection:nil];
[selfsetData:nil];
[selfsetResponse:nil];
}