2015-11-04-Android手動緩存js,css和瀏覽器默認緩存的優劣勢對比.md

爲什麼用緩存

將html,js和css緩存到localStorage,可以減少Http請求,從而優化頁面加載時間。

手動Web緩存

優勢

  • 可以緩存更多的內容到本地,包括大的圖片。

劣勢

  • 需要設計良好的緩存策略,並且客戶端和服務端都需要實現。
  • 當版本出現重大更新增加客戶端和服務器協作更新的複雜度。
  • 對於跨域請求問題,用JSONP實現,需要服務器和客戶端協商編寫處理方法。

WebView默認緩存

優勢

  • 設置方便,不用考慮交互問題。
  • 可以設置不同的緩存策略,降低服務器的負載。
  • 網上參考資料比較多,服務端可以通過標準配置來和客戶端交互。客戶端和服務器根據標準獨立開發,不需商定協作策略。

劣勢

  • 緩存容量不能設置太大,耗費流量較多的圖片緩存機制客戶端無法控制。

手動加載Web緩存實戰

  • 以jquery mobile爲例,在assets目錄裏面放入js,css,html資源文件。
    這裏寫圖片描述

  • 在寫本地html的時候引入assert裏的對應路徑

<!DOCTYPE html>  
<head>  
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />  
<meta name="viewport" content="width=device-width, initial-scale=1">   
<title> 標題 </title>  
<link rel="stylesheet" type="text/css" href="file:///android_asset/css/jquery.mobile-1.4.2.min.css">  
<script src="file:///android_asset/js/jquery-1.7.1.min.js"></script>  
<script src="file:///android_asset/js/jquery.mobile-1.4.2.min.js"></script>  
</head>  
<body>   

 <div data-role="page">  
   <div data-role="header">  
    <h1>My Title</h1>  
   </div>  

   <div data-role="content">  
    <ul data-role="listview" data-inset="true" >  
      <li><a href="#">Acura</a></li>  
      <li><a href="#">Audi</a></li>  
      <li><a href="#">BMW</a></li>  
      <li><a href="#">Cadillac</a></li>  
      <li><a href="#">Ferrari</a></li>  
    </ul>   
   </div>  
 </div><!-- /page -->  

</body>  
</html>
  • 在代碼裏訪問頁面
public class MainActivity extends Activity {

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    WebView webview = (WebView)findViewById(R.id.webView1);
    WebSettings wv_setttig = webview.getSettings();
    wv_setttig.setJavaScriptEnabled(true);

    String url = "file:///android_asset/index.html";
    webview.loadUrl(url);

  }

}
  • 最後效果如下:
    這裏寫圖片描述

打開WebView內置緩存機制實戰

當我們加載Html時候,會在我們data/應用package下生成database與cache兩個文件夾:
我們請求的Url記錄是保存在webviewCache.db裏,而url的內容是保存在webviewCache文件夾下.
WebView中存在着兩種緩存:網頁數據緩存(存儲打開過的頁面及資源)、H5緩存(即AppCache)。

網頁緩存

緩存構成

/data/data/package_name/cache/
/data/data/package_name/database/webview.db
/data/data/package_name/database/webviewCache.db
綜合可以得知 webview 會將我們瀏覽過的網頁url已經網頁文件(css、圖片、js等)保存到數據庫表中

緩存模式(5種)

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,都使用緩存中的數據。
如:www.taobao.com的cache-control爲no-cache,在模式LOAD_DEFAULT下,無論如何都會從網絡上取數據,如果沒有網絡,就會出現錯誤頁面;在LOAD_CACHE_ELSE_NETWORK模式下,無論是否有網絡,只要本地有緩存,都使用緩存。本地沒有緩存時才從網絡上獲取。
www.360.com.cn的cache-control爲max-age=60,在兩種模式下都使用本地緩存數據。

總結:根據以上兩種模式,建議緩存策略爲,判斷是否有網絡,有的話,使用LOAD_DEFAULT,無網絡時,使用LOAD_CACHE_ELSE_NETWORK。

設置WebView 緩存模式

private void initWebView() { 

        mWebView.getSettings().setJavaScriptEnabled(true); 
        mWebView.getSettings().setRenderPriority(RenderPriority.HIGH); 
        mWebView.getSettings().setCacheMode(WebSettings.LOAD_DEFAULT);  //設置 緩存模式 
        // 開啓 DOM storage API 功能 
        mWebView.getSettings().setDomStorageEnabled(true); 
        //開啓 database storage API 功能 
        mWebView.getSettings().setDatabaseEnabled(true);  
        String cacheDirPath = getFilesDir().getAbsolutePath()+APP_CACAHE_DIRNAME; 
//      String cacheDirPath = getCacheDir().getAbsolutePath()+Constant.APP_DB_DIRNAME; 
        Log.i(TAG, "cacheDirPath="+cacheDirPath); 
        //設置數據庫緩存路徑 
        mWebView.getSettings().setDatabasePath(cacheDirPath); 
        //設置  Application Caches 緩存目錄 
        mWebView.getSettings().setAppCachePath(cacheDirPath); 
        //開啓 Application Caches 功能 
        mWebView.getSettings().setAppCacheEnabled(true); 
    }

清除緩存

/**
     * 清除WebView緩存
     */ 
    public void clearWebViewCache(){ 

        //清理Webview緩存數據庫 
        try { 
            deleteDatabase("webview.db");  
            deleteDatabase("webviewCache.db"); 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 

        //WebView 緩存文件 
        File appCacheDir = new File(getFilesDir().getAbsolutePath()+APP_CACAHE_DIRNAME); 
        Log.e(TAG, "appCacheDir path="+appCacheDir.getAbsolutePath()); 

        File webviewCacheDir = new File(getCacheDir().getAbsolutePath()+"/webviewCache"); 
        Log.e(TAG, "webviewCacheDir path="+webviewCacheDir.getAbsolutePath()); 

        //刪除webview 緩存目錄 
        if(webviewCacheDir.exists()){ 
            deleteFile(webviewCacheDir); 
        } 
        //刪除webview 緩存 緩存目錄 
        if(appCacheDir.exists()){ 
            deleteFile(appCacheDir); 
        } 
    }

完整代碼

package com.example.webviewtest; 

import java.io.File; 

import android.app.Activity; 
import android.graphics.Bitmap; 
import android.os.Bundle; 
import android.util.Log; 
import android.view.View; 
import android.webkit.JsPromptResult; 
import android.webkit.JsResult; 
import android.webkit.WebChromeClient; 
import android.webkit.WebSettings; 
import android.webkit.WebSettings.RenderPriority; 
import android.webkit.WebView; 
import android.webkit.WebViewClient; 
import android.widget.RelativeLayout; 
import android.widget.TextView; 
import android.widget.Toast; 

public class MainActivity extends Activity { 

    private static final String TAG = MainActivity.class.getSimpleName(); 
    private static final String APP_CACAHE_DIRNAME = "/webcache"; 
    private TextView tv_topbar_title; 
    private RelativeLayout rl_loading; 
    private WebView mWebView; 
    private String url; 

    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState); 
        setContentView(R.layout.activity_main); 

        //url:http://m.dianhua.cn/detail/31ccb426119d3c9eaa794df686c58636121d38bc?apikey=jFaWGVHdFVhekZYWTBWV1ZHSkZOVlJWY&app=com.yulore.yellowsdk_ios&uid=355136051337627 
        url = "http://m.dianhua.cn/detail/31ccb426119d3c9eaa794df686c58636121d38bc?apikey=jFaWGVHdFVhekZYWTBWV1ZHSkZOVlJWY&app=com.yulore.yellowsdk_ios&uid=355136051337627"; 
        findView(); 
    } 

    private void findView() { 

        tv_topbar_title = (TextView) findViewById(R.id.tv_topbar_title); 

        rl_loading = (RelativeLayout) findViewById(R.id.rl_loading); 

        mWebView = (WebView) findViewById(R.id.mWebView); 

        initWebView(); 

        mWebView.setWebViewClient(new WebViewClient() { 

            @Override 
            public void onLoadResource(WebView view, String url) { 

                Log.i(TAG, "onLoadResource url="+url); 

                super.onLoadResource(view, url); 
            } 

            @Override 
            public boolean shouldOverrideUrlLoading(WebView webview, String url) { 

                Log.i(TAG, "intercept url="+url); 

                webview.loadUrl(url); 

                return true; 
            } 

            @Override 
            public void onPageStarted(WebView view, String url, Bitmap favicon) { 

                Log.e(TAG, "onPageStarted"); 

                rl_loading.setVisibility(View.VISIBLE); // 顯示加載界面 
            } 

            @Override 
            public void onPageFinished(WebView view, String url) { 

                String title = view.getTitle(); 

                Log.e(TAG, "onPageFinished WebView title=" + title); 

                tv_topbar_title.setText(title); 
                tv_topbar_title.setVisibility(View.VISIBLE); 

                rl_loading.setVisibility(View.GONE); // 隱藏加載界面 
            } 

            @Override 
            public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) { 

                rl_loading.setVisibility(View.GONE); // 隱藏加載界面 

                Toast.makeText(getApplicationContext(), "", 
                        Toast.LENGTH_LONG).show(); 
            } 
        }); 

        mWebView.setWebChromeClient(new WebChromeClient() { 

            @Override 
            public boolean onJsAlert(WebView view, String url, String message, JsResult result) { 

                Log.e(TAG, "onJsAlert " + message); 

                Toast.makeText(getApplicationContext(), message, Toast.LENGTH_SHORT).show(); 

                result.confirm(); 

                return true; 
            } 

            @Override 
            public boolean onJsConfirm(WebView view, String url, String message, JsResult result) { 

                Log.e(TAG, "onJsConfirm " + message); 

                return super.onJsConfirm(view, url, message, result); 
            } 

            @Override 
            public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) { 

                Log.e(TAG, "onJsPrompt " + url); 

                return super.onJsPrompt(view, url, message, defaultValue, result); 
            } 
        }); 

        mWebView.loadUrl(url); 
    } 

    private void initWebView() { 

        mWebView.getSettings().setJavaScriptEnabled(true); 
        mWebView.getSettings().setRenderPriority(RenderPriority.HIGH); 
        mWebView.getSettings().setCacheMode(WebSettings.LOAD_DEFAULT);  //設置 緩存模式 
        // 開啓 DOM storage API 功能 
        mWebView.getSettings().setDomStorageEnabled(true); 
        //開啓 database storage API 功能 
        mWebView.getSettings().setDatabaseEnabled(true);  
        String cacheDirPath = getFilesDir().getAbsolutePath()+APP_CACAHE_DIRNAME; 
//      String cacheDirPath = getCacheDir().getAbsolutePath()+Constant.APP_DB_DIRNAME; 
        Log.i(TAG, "cacheDirPath="+cacheDirPath); 
        //設置數據庫緩存路徑 
        mWebView.getSettings().setDatabasePath(cacheDirPath); 
        //設置  Application Caches 緩存目錄 
        mWebView.getSettings().setAppCachePath(cacheDirPath); 
        //開啓 Application Caches 功能 
        mWebView.getSettings().setAppCacheEnabled(true); 
    } 

    /**
     * 清除WebView緩存
     */ 
    public void clearWebViewCache(){ 

        //清理Webview緩存數據庫 
        try { 
            deleteDatabase("webview.db");  
            deleteDatabase("webviewCache.db"); 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 

        //WebView 緩存文件 
        File appCacheDir = new File(getFilesDir().getAbsolutePath()+APP_CACAHE_DIRNAME); 
        Log.e(TAG, "appCacheDir path="+appCacheDir.getAbsolutePath()); 

        File webviewCacheDir = new File(getCacheDir().getAbsolutePath()+"/webviewCache"); 
        Log.e(TAG, "webviewCacheDir path="+webviewCacheDir.getAbsolutePath()); 

        //刪除webview 緩存目錄 
        if(webviewCacheDir.exists()){ 
            deleteFile(webviewCacheDir); 
        } 
        //刪除webview 緩存 緩存目錄 
        if(appCacheDir.exists()){ 
            deleteFile(appCacheDir); 
        } 
    } 

    /**
     * 遞歸刪除 文件/文件夾
     * 
     * @param file
     */ 
    public void deleteFile(File file) { 

        Log.i(TAG, "delete file path=" + file.getAbsolutePath()); 

        if (file.exists()) { 
            if (file.isFile()) { 
                file.delete(); 
            } else if (file.isDirectory()) { 
                File files[] = file.listFiles(); 
                for (int i = 0; i < files.length; i++) { 
                    deleteFile(files[i]); 
                } 
            } 
            file.delete(); 
        } else { 
            Log.e(TAG, "delete file no exists " + file.getAbsolutePath()); 
        } 
    } 

}
發佈了39 篇原創文章 · 獲贊 3 · 訪問量 5萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章