缓存webview图片资源的两种方式

在使用iOS的webview的时候发现这样一个问题,加载一个网页进来,webview会负责缓存页面里的css,js和图片这些资源。但是这个缓存不受开发者控制,缓存时间非常短.所以为了节省用户流量,大量的Hybird混合应用和电商类应用都在研究H5页面热更新和图片交由本地保存的策略,今天我们来研究一下如何缓存webivew的图片资源。

第一种方式:NSURLCache

作为iOS御用的缓存类,NSURLCache给我们提供了一个简单的缓存实现方式,但在使用的时候,某些情况下,应用中的系统组件会将缓存的内存容量设为0MB,这就禁用了缓存。解决这个行为的一种方式就是通过自己的实现子类化NSURLCache,拒绝将内存缓存大小设为0。
在我们仅仅为了实现一个缓存图片的类的时候,我们的代码极其简单,就是继承NSURLCache,重载下面这两个方法,就实现类图片缓存和读取:

- (NSCachedURLResponse *)cachedResponseForRequest:(NSURLRequest *)request {
                                                                                                                                                                                                                          
    NSString *pathString = [[request URL] absoluteString];
                                                                                                                                                                                                                          
    if(![pathString hasSuffix:@".jpg"] || ![pathString hasSuffix:@".png"]) {
        return[super cachedResponseForRequest:request];
    }
                                                                                                                                                                                                                          
    if([[BGURLCache sharedCache] hasDataForURL:pathString]) {
        NSData *data = [[BGURLCache sharedCache] dataForURL:pathString];
        NSURLResponse *response = [[NSURLResponse alloc] initWithURL:[request URL]
                                                             MIMEType:[pathString hasSuffix:@".jpg"]?@"image/jpg":@"image/png"
                                                expectedContentLength:[data length]
                                                     textEncodingName:nil];
        return  [[NSCachedURLResponse alloc] initWithResponse:response data:data];        
    }
    return[super cachedResponseForRequest:request];
}
                                                                                                                                                                                                                      
- (void)storeCachedResponse:(NSCachedURLResponse *)cachedResponse forRequest:(NSURLRequest *)request {
    NSString *pathString = [[request URL] absoluteString];
    if(![pathString hasSuffix:@".jpg"] || ![pathString hasSuffix:@".png"]) {
        [super storeCachedResponse:cachedResponse forRequest:request];
        return;
    }
                                                                                                                                                                                                                          
    [[BGURLCache sharedCache] storeData:cachedResponse.data forURL:pathString];
}

NSURLCache的坑

使用NSURLCache做缓存看起来简单又好用,但是为什么各种大佬都不建议用呢?具体到我这个需求,是因为下面这几个坑:
1.只能用在get请求里面,post可以洗洗睡了。
2.需要服务器定义数据是否发生变化,需要在请求头里查找是否修改了的信息。公司服务器没有定义的话,就不能够判断读取的缓存数据是否需要刷新。
3.删除缓存的removeCachedResponseForRequest 这个方法是无效的.所以缓存是不会被删除的—只有删除全部缓存才有效。

总结

不能删除对应的缓存方案是没有意义的,所以我放弃了这个方案。

第二种方案:NSURLProtocol

NSURLProtocol或许是URL加载系统中最功能强大但同时也是最晦涩的部分了。它是一个抽象类,你可以通过子类化来定义新的或已经存在的URL加载行为。还好我们的需求只是做一个图片的缓存需求,不然就抓瞎了,在具体到这个类的时候我们要做的也很简单,拦截图片加载请求,转为从本地文件加载。
1.我们要认识NSURLProtocol,首先它是一个抽象类,不能够直接使用必须被子类化之后才能使用。子类化 NSURLProtocol 的第一个任务就是告诉它要控制什么类型的网络请求。比如说如果你想要当本地有资源的时候请求直接使用本地资源文件,那么相关的请求应该对应已有资源的文件名。
这部分逻辑定义在

+ (BOOL)canInitWithRequest:(NSURLRequest *)request;

中,如果返回 YES,该请求就会被其控制。返回 NO 则直接跳入下一个Protocol,一句话,我们可以在里面完成拦截图片加载请求,转为从本地文件加载的大概逻辑。
2.获取和设置一个请求对象的当前状态,可以在Protocol的各种方法中传递当前request我们自定义的状态。核心方法是:

+ (nullable id)propertyForKey:(NSString *)key inRequest:(NSURLRequest *)request;
+ (void)setProperty:(id)value forKey:(NSString *)key inRequest:(NSMutableURLRequest *)request;
+ (void)removePropertyForKey:(NSString *)key inRequest:(NSMutableURLRequest *)request;

3.最最重要的方法

 -startLoading 
 -stopLoading

不同的自定义子类在调用这两个方法是会传入不同的内容,但共同点都是要围绕protocol的client属性进行操作,在对-startLoading 和-stopLoading的实现中,需要在恰当的时候让client调用每一个delegate方法。我们在startloading中初始化NSURLSessionDataTask,在session的代理方法中传递数据给client的代理方法。在stoploading中结束当前datatask。
4.向系统注册该NSURLProtocol,当请求被加载时,系统会向每一个注册过的protocol询问是否能控制该请求,第一个通过+canInitWithRequest: 回答为 YES 的protocol就会控制该请求。URLProtocol会被以注册顺序的反序访问,所以当在 -application:didFinishLoadingWithOptions:方法中调用 [NSURLProtocol registerClass:[BGURLProtocol class]]; 时,你自己写的protocol比其他内建的protocol拥有更高的优先级。
4.核心代码
BGURLProtocol.m:

+ (BOOL)canInitWithRequest:(NSURLRequest *)request
{
    NSString *urlString = request.URL.absoluteString;
    NSString* extension = request.URL.pathExtension;
    if([NSURLProtocol propertyForKey:@"ProtocolHandledKey" inRequest:request]) {
        return NO;
    }
    BOOL isImage = [@[@"png", @"jpeg", @"gif", @"jpg"] indexOfObjectPassingTest:^BOOL(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        return [extension compare:obj options:NSCaseInsensitiveSearch] == NSOrderedSame;
    }] != NSNotFound;
    if (isImage)
    {
        NSString *filePath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
        filePath = [filePath stringByAppendingPathComponent:[[urlString componentsSeparatedByString:@"/"] lastObject]];
        if ([[NSFileManager defaultManager] fileExistsAtPath:filePath])
        {
            return YES;
        }
        else
        {
            static NSInteger requestCount = 0;
            static NSInteger requestRefresh = 0;
            NSMutableURLRequest *newRequest = [request mutableCopy];
            [NSURLProtocol setProperty:@YES forKey:@"ProtocolHandledKey" inRequest:newRequest];
            NSString *url = [@"http://www.baidu.com/img/" stringByAppendingString:[[urlString componentsSeparatedByString:@"/"] lastObject]];
            requestCount++;
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                requestRefresh = 1;
                if (requestCount == 0)
                {
                    [[NSNotificationCenter defaultCenter] postNotificationName:kImageDownloadNotification object:[[urlString componentsSeparatedByString:@"/"] lastObject]];
                }
            });
            [[SDWebImageDownloader sharedDownloader] downloadImageWithURL:[NSURL URLWithString:url] options:SDWebImageDownloaderUseNSURLCache progress:^(NSInteger receivedSize, NSInteger expectedSize) {
                
            } completed:^(UIImage *image, NSData *data, NSError *error, BOOL finished) {
                requestCount--;
                NSString *filePath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
                filePath = [filePath stringByAppendingPathComponent:[[urlString componentsSeparatedByString:@"/"] lastObject]];
                [data writeToFile:filePath atomically:YES];
                if (requestCount == 0 && requestRefresh == 1)
                {
                    [[NSNotificationCenter defaultCenter] postNotificationName:kImageDownloadNotification object:[[urlString componentsSeparatedByString:@"/"] lastObject]];
                }
                
            }];
        }
        
        return YES;
    }else{
        return YES;
    }
}
-(void)startLoading
{
    NSMutableURLRequest *newRequest = [self.request mutableCopy];
    [NSURLProtocol setProperty:@YES forKey:@"ProtocolHandledKey" inRequest:newRequest];
    NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                                                          delegate:self
                                                     delegateQueue:[[NSOperationQueue alloc] init]];

    self.connection = [session dataTaskWithRequest:newRequest];
     [self.connection resume];
}

- (void)stopLoading {
    [self.connection cancel];
    self.connection =nil;
}

-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data{
    [self.client URLProtocol:self didLoadData:data];
}

- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
didReceiveResponse:(NSURLResponse *)response
 completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler{
    if ([response isKindOfClass:[NSHTTPURLResponse class]]) {
        NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *) response;
        NSURLResponse *retResponse = [[NSHTTPURLResponse alloc] initWithURL:[NSURL URLWithString:response.URL.absoluteString] statusCode:httpResponse.statusCode HTTPVersion:(__bridge NSString *)kCFHTTPVersion1_1 headerFields:httpResponse.allHeaderFields];
        [self.client URLProtocol:self didReceiveResponse:retResponse cacheStoragePolicy:NSURLCacheStorageNotAllowed];
    } else {
        NSURLResponse *retResponse = [[NSURLResponse alloc] initWithURL:[NSURL URLWithString:response.URL.absoluteString] MIMEType:response.MIMEType expectedContentLength:response.expectedContentLength textEncodingName:response.textEncodingName];
        [self.client URLProtocol:self didReceiveResponse:retResponse cacheStoragePolicy:NSURLCacheStorageNotAllowed];
    }
    completionHandler(NSURLSessionResponseAllow);
    
}

- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    // 请求完成,成功或者失败的处理
    if (error) {
        [self.client URLProtocol:self didFailWithError:error];
    }else{
        [self.client URLProtocolDidFinishLoading:self];
    }
}

webview初始化:

    NSString *htmlString = [NSString stringWithContentsOfURL:[NSURL URLWithString:@"http://www.baidu.com"] encoding:NSUTF8StringEncoding error:nil];
    
    htmlString = [htmlString stringByReplacingOccurrencesOfString:@"//www.baidu.com/img/" withString:@""];
    _htmlString = htmlString;
    NSString *baseUrl = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
    [webView loadHTMLString:htmlString baseURL:[NSURL fileURLWithPath:baseUrl]];

5.我们现在实现了用NSURLProtocol拦截.png和.jpg的网络请求,让UIWebView本身的图片下载发不出去,拦截的链接通过SDWebImage下载资源到本地目录,用WebView的loadHTMLString:baseURL:方法来实现读取本地目录的图片显示,当下载图片超过2秒,并且请求数为0时发送通知给webView刷新显示本地资源。但是,现在已经iOS11了啊,UIWebView内存占用太大已经跟不上时代了,WKWebiview默认不支持NSURLProtocol,所以我们得找个办法让WK支持我们子类话的protocol。所以我找到了这段:

//Class cls = NSClassFromString(@"WKBrowsingContextController");
Class cls = [[[WKWebView new] valueForKey:@"browsingContextController"] class];
SEL sel = NSSelectorFromString(@"registerSchemeForCustomProtocol:");
if ([(id)cls respondsToSelector:sel]) {    
// 把 http 和 https 请求交给 NSURLProtocol 处理  
        [(id)cls performSelector:sel withObject:@"http"];   
        [(id)cls performSelector:sel withObject:@"https"];
}

这样,我们就完成了webview缓存图片资源的需求。
参考资料:
http://bbs.csdn.net/topics/390831054
http://blog.csdn.net/jason_chen13/article/details/51984823
https://github.com/Yeatse/NSURLProtocol-WebKitSupport
http://blog.csdn.net/xanxus46/article/details/51946432
http://blog.csdn.net/u011661836/article/details/70241061

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,056评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,842评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,938评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,296评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,292评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,413评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,824评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,493评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,686评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,502评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,553评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,281评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,820评论 3 305
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,873评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,109评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,699评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,257评论 2 341

推荐阅读更多精彩内容