您的位置:首页 > 移动开发 > Android开发

Android webView 缓存 Cache + HTML5离线功能解决

2016-09-19 10:56 537 查看
WebView的缓存可以分为页面缓存和数据缓存。

           页面缓存是指加载一个网页时的html、JS、CSS等页面或者资源数据。这些缓存资源是由于浏览器的行为而产生,开发者只能通过配置HTTP响应头影响浏览器的行为才能间接地影响到这些缓存数据。

         他们的索引存放在/data/data/package_name/databases下。他们的文件存放在/data/data/package_name/cache/xxxwebviewcachexxx下。文件夹的名字在2.x和4.x上有所不同,但都文件夹名字中都包含webviewcache。

            数据缓存分为两种:AppCache和DOM Storage(Web Storage)。他们是因为页面开发者的直接行为而产生。所有的缓存数据都由开发者直接完全地掌控。
AppCache使我们能够有选择的缓冲web浏览器中所有的东西,从页面、图片到脚本、css等等。尤其在涉及到应用于网站的多个页面上的CSS和JavaScript文件的时候非常有用。其大小目前通常是5M。
            在Android上需要手动开启(setAppCacheEnabled),并设置路径(setAppCachePath)和容量(setAppCacheMaxSize)

Android中Webkit使用一个db文件来保存AppCache数据(my_path/ApplicationCache.db)

          如果需要存储一些简单的用key/value对即可解决的数据,DOM Storage是非常完美的方案。根据作用范围的不同,有Session Storage和Local Storage两种,分别用于会话级别的存储(页面关闭即消失)和本地化存储(除非主动删除,否则数据永远不会过期)。

在Android中可以手动开启DOM Storage(setDomStorageEnabled),设置存储路径(setDatabasePath)

Android中Webkit会为DOM Storage产生两个文件(my_path/localstorage/http_h5.m.taobao.com_0.localstorage和my_path/localstorage/Databases.db)

           另外,在Android中清除缓存时,如果需要清除Local Storage的话,仅仅删除Local Storage的本地存储文件是不够的,内存里面有缓存数据。如果再次进入页面,Local Storage中的缓存数据同样存在。需要杀死程序运行的当前进程再重新启动才可以。

           

---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

HTML5的离线应用功能可以使得WebApp即使在网络断开的情况下仍能正常使用,这是个非常有用的功能。近来工作中也要用到HTML5离线应用功能,由于是在Android平台上做,所以自然而然的选择Webview来解析网页。但如何使Webivew支持HTML5离线应用功能呢,经过反复摸索和上网查找资料,反复做试验终于成功了。

首先需配置webview的的一些属性,假设activity中已经有了一个Webview的实例对象,名为m_webview,然后增加以下代码:

[html] view
plaincopy

WebSettings webseting = m_webview.getSettings();  

    webseting.setDomStorageEnabled(true);             

        webseting.setAppCacheMaxSize(1024*1024*8);//设置缓冲大小,我设的是8M  

    String appCacheDir = this.getApplicationContext().getDir("cache", Context.MODE_PRIVATE).getPath();      

        webseting.setAppCachePath(appCacheDir);  

        webseting.setAllowFileAccess(true);  

        webseting.setAppCacheEnabled(true);  

        webseting.setCacheMode(WebSettings.LOAD_DEFAULT);   

webview可以设置一个WebChromeClient对象,在其onReachedMaxAppCacheSize函数对扩充缓冲做出响应。代码如下:

[html] view
plaincopy

m_webview.setWebChromeClient(m_chromeClient);  

    private WebChromeClient m_chromeClient = new WebChromeClient(){  

        //扩充缓存的容量    

    @Override  

    public void onReachedMaxAppCacheSize(long spaceNeeded,    

                long totalUsedQuota, WebStorage.QuotaUpdater quotaUpdater) {    

            quotaUpdater.updateQuota(spaceNeeded * 2);    

        }         

    };  

 

其次要修改http服务器中的配置,使其支持text/cache-manifest,我使用的是apache服务器,是windows版本的,在apache的conf文件夹中找到mime.types文件,打开后在文件的最后加上

“text/cache-manifest              mf  manifest”,重启服务器即可。这一步很重要,我就是因为服务器端没有配置这个,所以失败了好多次,最后是在附录链接1的回复中找到的线索。

经过以上设置Webview就可以支持HTML5的离线应用了。

附录链接1中说缓冲目录应该是getApplicationContext().getCacheDir().getAbsolutePath();但我经过试验后发现设置那个目录不起作用,可能是Android版本不同吧,我的是Android4.0.3,而他的可能是以前的Android版本吧。

缓冲目录使用getApplicationContext().getDir("cache", Context.MODE_PRIVATE).getPath()是从附录链接2中找到的线索。

附录链接:

1.http://alex.tapmania.org/2010/11/html5-cache-android-webview.html

2.http://johncookie.iteye.com/blog/1182459

3.HTML5 Offline官方文档:http://www.w3.org/TR/html5/offline.html#manifests

原因:

webview加载 服务端的网页,为了减少访问压力,用html5缓存技术,本地建了数据库,在手机浏览器里  可以显示页面,换成webView就不行了。

解决范例:

Activity code:

?
代码片段,双击复制
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public
 
class
 
efan_NewsReader 
extends
Activity {

        
/** Called when the activity is first created. */

        
@Override

        
public
void
onCreate(Bundle savedInstanceState)

        
{   

                
super
.onCreate(savedInstanceState);   

                
setContentView(R.layout.main);            

                
WebView myWebView=(WebView)findViewById(R.id.my_webview);   

                
myWebView.setWebViewClient(
new
WebViewClient());   

                
WebSettings settings = myWebView.getSettings();

               
// 开启javascript设置

               
settings.setJavaScriptEnabled(
true
);  

               
// 设置可以使用localStorage

               
settings.setDomStorageEnabled(
true
);

               
// 应用可以有数据库

               
settings.setDatabaseEnabled(
true
);   

               
String dbPath =
this
.getApplicationContest().getDir(
"database"
,
Context.MODE_PRIVATE).getPath();

               
settings.setDatabasePath(dbPath);

               
// 应用可以有缓存

               
settings.setAppCacheEnabled(
true
);            

               
String appCaceDir =
this
.getApplicationContext().getDir(
"cache"
,
Context.MODE_PRIVATE).getPath();

               
settings.setAppCachePath(appCaceDir);

               
 
               
myWebView.loadUrl(
"http://10.10.35.47:8080/html5test/test.htm"
);

        
}

}


HTML5 page source code: 

?
代码片段,双击复制
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
    
<
html
manifest
=
"mymanifest.manifest"
>

    
<
head
>

<
meta
http-equiv
=
"Content-Type"
content
=
"text/html;
content="
no-cache"
charset
=
utf
-8" />

<
script
type
=
"text/javascript"
src
=
"js/jquery-1.6.1.min.js"
></
script
>

 
<
script
>

$(document).ready(function(){      

 
    
databaseTest();

});

 
function databaseTest(){

 
 
    
//open database

     
var db = openDatabase('mydb', '1.0', 'Test DB', 2 * 1024 * 1024);  

 
      
db.transaction(function (tx) {            

      
tx.executeSql('CREATE TABLE IF NOT EXISTS testHtml (id unique, contentText)');

      
tx.executeSql('INSERT INTO testHtml (contentText) VALUES ("insert data test!")');  

       
});  

 
     
db.transaction(function(tx){           

     
tx.executeSql('SELECT * FROM testHtml',[],function(tx,result){

            
var len=result.rows.length;

            
var msg = "<
p
>Found rows: " + len + "</
p
>";  

             
$("#testinfo").append(msg);

        
},null);

     
});   

 
 
}

 
</
script
>

 
 
</
head
>

<
body
>

    
<
div
>here is test info:</
div
>

    
<
div
id
=
"testinfo"
></
div
>

</
body
>


其他设置还有:
settings.setCacheMode(WebSettings.LOAD_DEFAULT);   // 默认使用缓存
settings.setAppCacheMaxSize(8*1024*1024);   //缓存最多可以有8M
settings.setAllowFileAccess(true);   // 可以读取文件缓存(manifest生效)

in WebChromeClient :

?
代码片段,双击复制
01
02
03
04
05
06
07
08
09
10
myWebView.setWebChromeClient(
new
WebChromeClient()

{   

        
@Override
   
        
public
void
onExceededDatabaseQuota(String url, String databaseIdentifier,

                       
long
currentQuota,
long
estimatedSize,
long
totalUsedQuota,

                       
WebStorage.QuotaUpdater quotaUpdater)   

        
{        

                
quotaUpdater.updateQuota(estimatedSize * 
2
);   

        
}

}


or:

?
代码片段,双击复制
01
02
03
04
05
06
07
08
09
10
myWebView.setWebChromeClient(
new
WebChromeClient()

{

    
// 扩充缓存的容量   

        
@Override
   
        
public
void
onReachedMaxAppCacheSize(
long
spaceNeeded,
long
totalUsedQuota,

                       
WebStorage.QuotaUpdater quotaUpdater)   

        
{        

                
quotaUpdater.updateQuota(spaceNeeded * 
2
);

    
}

}


按照范例,我成功的解决了我的问题,而且之前弹出框所出现的找不到数据(提示:underfine)也解决了,这个应该是当初数据库没设所引起的。

WebView中存在着两种缓存:网页数据缓存(存储打开过的页面及资源)、H5缓存(即appcache)。

一、网页缓存

1、缓存构成
/data/data/package_name/cache/
/data/data/package_name/database/webview.db
/data/data/package_name/database/webviewCache.db

2、缓存模式
较难理解的是以下两个模式:
LOAD_DEFAULT,根据cache-control决定是否从网络上取数据。
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
如:m.taobao.com的cache-control为no-cache,在模式LOAD_DEFAULT下,无论如何都会从网络上取数据,如果没有网络,就会出现错误页面;在LOAD_CACHE_ELSE_NETWORK模式下,无论是否有网络,只要本地有缓存,都使用缓存。本地没有缓存时才从网络上获取。
m.sina.com.cn的cache-control为max-age=60,在两种模式下都使用本地缓存数据。

总结:根据以上两种模式,建议缓存策略为,判断是否有网络,有的话,使用LOAD_DEFAULT,无网络时,使用LOAD_CACHE_ELSE_NETWORK。

3、清除缓存
clearCache(boolean)。
CacheManager.clear。高版本中需要调用隐藏API。

4、控制大小
无系统API支持。
可选方式:定时统计缓存大小、按时间顺序删除缓存。

二、H5缓存

1、缓存构成
根据setAppCachePath(String appCachePath)提供的路径,在H5使用缓存过程中生成的缓存文件。

2、缓存模式
无模式选择,通过setAppCacheEnabled(boolean flag)设置是否打开。默认关闭,即,H5的缓存无法使用。

3、清除缓存
找到调用setAppCachePath(String appCachePath)设置缓存的路径,把它下面的文件全部删除就OK了。

4、控制大小
通过setAppCacheMaxSize(long appCacheMaxSize)设置缓存最大容量,默认为Max Integer。
同时,可能通过覆盖WebChromeClient.onReachedMaxAppCacheSize(long requiredStorage, long quota, WebStorage.QuotaUpdater quotaUpdater)来设置缓存超过先前设置的最大容量时的策略。

三、参考网址

以下地址有关于H5缓存的一些内幕,如每个Application只调用一次WebSettings.setAppCachePath(),WebSettings.setAppCacheMaxSize()被忽略等一系列问题,需要仔细阅读和实验。 http://code.google.com/p/android/issues/detail?id=24180
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息