在现在的app中,越来越多的公司开始使用webview来进行一些活动页面的展示,甚至一些公司开始使用webview做为主要显示组件,把所有的内容都使用H5来呈现,这样一来,就对WebView的加载速度开始有越来越高的要求,我们要讨论的是webview的原本缓存机制所存在的弊端和如何复写WebView的缓存机制。
首先说一下webview的自带缓存机制的弊端:
webview的自带缓存机制是无差别缓存,也就是说,不管是页面,样式还是图片,都会缓存到本地,刷新webview的缓存一般分为以下几种:
LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
LOAD_DEFAULT: 根据cache-control决定是否从网络上取数据。
LOAD_CACHE_NORMAL: API level 17中已经废弃, 从API level 11开始作用同LOAD_DEFAULT模式
LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
反正不管以上几种具体情况如何,都肯定不是我们想要的。
我们想要的机制是:
1.缓存自己想要缓存的内容。
2.指定一个缓存策略,在需要的时候重新去服务器获取最新数据
于是我想到了以下方法
重写
首先需要新建一个类,继承WebViewClient:
然后实现
方法
shouldInterceptRequest方法会将所有页面的资源URL都一一列举出来,这样一来就好办了,我们似乎只需要缓存自己想要缓存的url就可以了。
然后事实是不是这样的呢?
接下来我们看下DVDUrlCache的实现:
DVDUrlCache主要做了这么几件事:
1.封装一个内部类CacheEntry,做一些基本信息存储
首先说一下webview的自带缓存机制的弊端:
webview的自带缓存机制是无差别缓存,也就是说,不管是页面,样式还是图片,都会缓存到本地,刷新webview的缓存一般分为以下几种:
LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
LOAD_DEFAULT: 根据cache-control决定是否从网络上取数据。
LOAD_CACHE_NORMAL: API level 17中已经废弃, 从API level 11开始作用同LOAD_DEFAULT模式
LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
反正不管以上几种具体情况如何,都肯定不是我们想要的。
我们想要的机制是:
1.缓存自己想要缓存的内容。
2.指定一个缓存策略,在需要的时候重新去服务器获取最新数据
于是我想到了以下方法
重写
WebViewClient.shouldInterceptRequest(WebView view,WebResourceRequest request)
方法
首先需要新建一个类,继承WebViewClient:
public class DVDWebViewClient extends WebViewClient
然后实现
public WebResourceResponse shouldInterceptRequest(WebView view,WebResourceRequest request)
方法
shouldInterceptRequest方法会将所有页面的资源URL都一一列举出来,这样一来就好办了,我们似乎只需要缓存自己想要缓存的url就可以了。
然后事实是不是这样的呢?
@Override
public WebResourceResponse shouldInterceptRequest(WebView view,
WebResourceRequest request) {
//获取本地的URL主域名
String curDomain = request.getUrl().getHost();
//这行LOG可以不看
if (BuildConfig.DEBUG) {
Log.d(LOG_TAG, "curDomain " + curDomain + " request headers " + request.getRequestHeaders());
for (Map.Entry<String, String> entry : request.getRequestHeaders().entrySet()) {
Log.d(LOG_TAG, "key=" + entry.getKey() + " #####value=" + entry.getValue() + "\n");
}
}
//取不到domain就直接返回,把接下俩的动作交给webview自己处理
if (curDomain == null || !isPicUrl(curDomain)) {
return null;
}
if (BuildConfig.DEBUG) {
Log.d(LOG_TAG, "shouldInterceptRequest url " + request.getUrl().toString());
}
//读取当前webview正准备加载URL资源
String url = request.getUrl().toString();
try {
//根据资源url获取一个你要缓存到本地的文件名,一般是URL的MD5
String resFileName = getResourcesFileName(url);
if (resFileName == null || "".equals(resFileName)) {
return null;
}
//这里是处理本地缓存的URL,缓存到本地,或者已经缓存过的就直接返回而不去网络进行加载
this.dvdUrlCache.register(url, getResourcesFileName(url),
request.getRequestHeaders().get("Accept"), "UTF-8", DVDUrlCache.ONE_MONTH);
return this.dvdUrlCache.load(url);
} catch (Exception e) {
Log.e(LOG_TAG, "", e);
}
return null;
}
接下来我们看下DVDUrlCache的实现:
DVDUrlCache主要做了这么几件事:
1.封装一个内部类CacheEntry,做一些基本信息存储
private static class CacheEntry {
//用作存储的URL
public String url;
//本地保存的文件名称
public String fileName;
//标记资源的头部,通过request参数取回
String mimeType;
//需要缓存的资源文件的编码
public String encoding;
//缓存最大有效时间
long maxAgeMillis;
private CacheEntry(String url, String fileName,
String mimeType, String encoding, long maxAgeMillis) {
this.url = url;
this.fileName = fileName;
this.mimeType = mimeType;
this.encoding = encoding;