2015-11-04-Android手动缓存js,css和浏览器默认缓存的优劣势对比.md

转载 2015年11月19日 15:46:19

为什么用缓存

将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()); 
        } 
    } 

}

Android WebView 缓存处理

当我们加载Html时候,会在我们data/应用package下生成database与cache两个文件夹: 我们请求的Url记录是保存在webviewCache.db里,而url的内容是保存在webv...
  • FX_SKY
  • FX_SKY
  • 2013年11月29日 15:05
  • 11707

【转载】----web网站css,js更新后客户浏览器缓存问题,需要刷新才能正常展示的解决办法

问题描述 最近将公司官网样式进行了调整,部署到服务器后访问发现页面展示不正常,但是刷新之后就会展示正常。 转载自:http://blog.csdn.net/csdn100861/article...

解决修改css或js文件,浏览器缓存更新问题。

在搜索引擎中搜索关键字.htaccess 缓存,你可以搜索到很多关于设置网站文件缓存的教程,通过设置可以将css、js等不太经常更新的文件缓存在浏览器端,这样访客每次访问你的网站的时候,浏览器就可以从...

关于引用JS和CSS刷新浏览器缓存问题

有时候我们会碰到上线的新版本都要刷新一次缓存的问题。那是因为改了JS的内容,但是JSP引用的地方后面的字符串未发生改变导致浏览器读取浏览器缓存而不会重新加载新的JS内容,以下提供两种解决方式: 1....

web网站css,js更新后客户浏览器缓存问题,需要刷新才能正常展示的解决办法

原文出处:http://blog.csdn.net/csdn100861/article/details/50684438 问题描述 最近将公司官网样式进行了调整,部署到服务器后访...

强制刷新浏览器调用的js和css以及js清除浏览器缓存的几种方法

今天遇到一个js缓存的问题,就是有一个公用的js里面添加了方法,但是在浏览器页面调用的时候怎么整都说取不到新加的方法,都是老的js文件,搞了浏览器自带的清除和电脑优化软件的清空都不得行,后来还是小组里...

HTTP+JS+浏览器缓存技术

  • 2016年12月02日 10:17
  • 122KB
  • 下载

摸爬滚打DirectX11_day04——顶点缓存介绍

在计算机所描绘的3D世界中,所有的物体模型都是通过多边形网格(三角形,四边形)来逼近表示的,之后在这些网格轮廓的表面贴上相应的图片。任何物体都可以用三角形网格来逼近表示。(这里的多边形不仅仅有顶点的坐...

2015-3-31 浏览器本地缓存问题

今天被一个问题困扰了,PM要做一个专题页,前两周让先做个敬请期待的页面,然后做了二维码指向此页面,印了海报贴到了公共场所,后面内容完善后又重新上线了,结果因服务器缓存问题没更新,后来让运维部门手动清理...
  • hao_fe
  • hao_fe
  • 2015年03月31日 13:41
  • 526

浏览器与HTTP网络协议缓存原理分析 转自网界网:http://news.cnw.com.cn/news-china/htm2015/20151027_322909.shtml

浏览器缓存原理 文字版描述 ①浏览器第一次访问服务器资源 /index.html 在浏览器中没有缓存文件,直接向服务器发送请求。 服务器返回 200 OK,实体中返回 inde...
  • n8765
  • n8765
  • 2015年10月28日 15:28
  • 942
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:2015-11-04-Android手动缓存js,css和浏览器默认缓存的优劣势对比.md
举报原因:
原因补充:

(最多只允许输入30个字)