Android webView 緩存 Cache + HTML5離線功能 解決


時間 2013-06-11 21:06:14  CSDN博客
原文  http://blog.csdn.net/moubenmao/article/details/9076917
主題 Android HTML5
            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 plain copy
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 plain copy
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              


經過以上設置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

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: 
?
代碼片段,雙擊復制

    < 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 : 
?
代碼片段,雙擊復制

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: 
?
代碼片段,雙擊復制

10
myWebView.setWebChromeClient( new WebChromeClient()
{
    // 擴充緩存的容量   
        @Override    
        public void onReachedMaxAppCacheSize( long spaceNeeded, long totalUsedQuota,
                        WebStorage.QuotaUpdater quotaUpdater)   
        {        
                quotaUpdater.updateQuota(spaceNeeded * 2 );
    }
}

按照范例,我成功的解決了我的問題,而且之前彈出框所出現的找不到數據(提示:underfine)也解決了,這個應該是當初數據庫沒設所引起的。 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM