Android之 -WebView实现离线缓存阅读

前言

本篇博客要实现的是一个离线下载和离线阅读的功能,这是很多阅读类app都常见的一个功能,典型的应用就是网易新闻。什么是离线下载?其实这个概念是比较模糊,是离线之后下载呢,还是下载之后离线,但稍微有点脑子的人都知道没有网络之后怎么下载呢?所以离线下载这个功能是”在有网络的情况下,把资源下载到本地“,离线阅读就是”在没有网络或者网络不好的时候,阅读本地好缓存的文章资源“。这样就很清楚我们要的这两个具体的功能需求了。

实现思路

小巫这里提供两个实现思路,一个就是自己写逻辑,一个是通过WebView本身自带的缓存功能来实现。 先来说第一个思路:

  1. 定义一个离线下载的服务Service
  2. 启动后台服务Service来执行异步下载
  3. 存储到本地数据库中
  4. 每一次加载url之前,先判断数据库是否存在缓存内容 5.如果存在缓存,优先加载本地缓存,如果不存在,才执行联网请求

第二个思路就是本篇 博客要讲的通过WebView自带的缓存功能来实现离线阅读 这里小巫接着上一篇博客的例子来添加代码。

参考1:http://www.open-open.com/lib/view/open1392188052301.html 参考2:http://87426628.blog.163.com/blog/static/6069361820139183417725/

Android WebView缓存可以分为页面缓存和数据缓存 页面缓存是指加载一个网页时html、JS、CSS等页面或者资源数据。 数据缓存分别为两种:AppCache和DOM Storage(Web Storage)。 AppCache也是我们的H5缓存,我们可以设置缓存的目录 Dom Storage具有Session Storage和Local Storage两种,前者是会话级别的存储,页面关闭之后就消失了,后者是本地化存储。 如果我们手机有root的权限的话,就可以看到/data/data/package_name/下的文件目录,我们就会发现webview为我们创建 app_webview,这个应该就是webview的缓存目录的位置。

public void initWebView() {
    mWebView.getSettings().setJavaScriptEnabled(true);
    mWebView.getSettings().setRenderPriority(RenderPriority.HIGH);
    // 建议缓存策略为,判断是否有网络,有的话,使用LOAD_DEFAULT,无网络时,使用LOAD_CACHE_ELSE_NETWORK

    mWebView.getSettings().setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK); // 设置缓存模式
    // 开启DOM storage API 功能
    mWebView.getSettings().setDomStorageEnabled(true);
    // 开启database storage API功能
    mWebView.getSettings().setDatabaseEnabled(true);
    String cacheDirPath = getFilesDir().getAbsolutePath()
            + APP_CACHE_DIRNAME;
    Log.i("cachePath", cacheDirPath);
    // 设置数据库缓存路径
    mWebView.getSettings().setDatabasePath(cacheDirPath); // API 19 deprecated
    // 设置Application caches缓存目录
    mWebView.getSettings().setAppCachePath(cacheDirPath);
    // 开启Application Cache功能
    mWebView.getSettings().setAppCacheEnabled(true);

    Log.i("databasepath", mWebView.getSettings().getDatabasePath());

}

这里加载我博客里的一篇文章:

public void findView() {
    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 view, String url) {
            Log.i(TAG, "intercept url=" + url);
            view.loadUrl(url);
            return true;
        }

        // 页面开始时调用
        @Override
        public void onPageStarted(WebView view, String url, Bitmap favicon) {
            Log.e(TAG, "onPageStarted");
            super.onPageStarted(view, url, favicon);
        }

        // 页面加载完成调用
        @Override
        public void onPageFinished(WebView view, String url) {
            super.onPageFinished(view, url);
        }

        @Override
        public void onReceivedError(WebView view, int errorCode,
                String description, String failingUrl) {
            super.onReceivedError(view, errorCode, description, failingUrl);
            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 super.onJsAlert(view, url, message, result);
        }

        @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);
}

完整代码

package com.infzm.webview;

import java.io.File;

import android.app.Activity; import android.content.Intent; import android.graphics.Bitmap; import android.net.Uri; import android.os.Bundle; import android.util.Log; import android.view.KeyEvent; import android.view.View; import android.view.View.OnClickListener; 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.Button; import android.widget.Toast;

public class MainActivity extends Activity implements OnClickListener {

private WebView mWebView;
private Button nightModeBtn;
private Button lightModeBtn;

// -----
private static final String TAG = MainActivity.class.getSimpleName();
private static final String APP_CACHE_DIRNAME = "/webcache"; // web缓存目录
private String url; // 网页url

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

    mWebView = (WebView) this.findViewById(R.id.webview);
    nightModeBtn = (Button) this.findViewById(R.id.btn_nightmode);
    lightModeBtn = (Button) this.findViewById(R.id.btn_lightmode);

    nightModeBtn.setOnClickListener(this);
    lightModeBtn.setOnClickListener(this);

    // webView.loadUrl("http://www.baidu.com");

    WebSettings settings = mWebView.getSettings();
    // 设置javaScript可用
    settings.setJavaScriptEnabled(true);
    // 绑定javaScript接口,可以实现在javaScript中调用我们的Android代码
    // webView.addJavascriptInterface(new WebAppInterface(this), "Android");
    // webView.setWebViewClient(new MyWebViewClient());

    // 加载assets目录下的html页面
    // mWebView.loadUrl("file:///android_asset/01.html");
    url = "http://blog.csdn.net/wwj_748/article/details/44810283";

    findView();
}

public void findView() {
    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 view, String url) {
            Log.i(TAG, "intercept url=" + url);
            view.loadUrl(url);
            return true;
        }

        // 页面开始时调用
        @Override
        public void onPageStarted(WebView view, String url, Bitmap favicon) {
            Log.e(TAG, "onPageStarted");
            super.onPageStarted(view, url, favicon);
        }

        // 页面加载完成调用
        @Override
        public void onPageFinished(WebView view, String url) {
            super.onPageFinished(view, url);
        }

        @Override
        public void onReceivedError(WebView view, int errorCode,
                String description, String failingUrl) {
            super.onReceivedError(view, errorCode, description, failingUrl);
            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 super.onJsAlert(view, url, message, result);
        }

        @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);
}

public void initWebView() {
    mWebView.getSettings().setJavaScriptEnabled(true);
    mWebView.getSettings().setRenderPriority(RenderPriority.HIGH);
    // 建议缓存策略为,判断是否有网络,有的话,使用LOAD_DEFAULT,无网络时,使用LOAD_CACHE_ELSE_NETWORK

    mWebView.getSettings().setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK); // 设置缓存模式
    // 开启DOM storage API 功能
    mWebView.getSettings().setDomStorageEnabled(true);
    // 开启database storage API功能
    mWebView.getSettings().setDatabaseEnabled(true);
    String cacheDirPath = getFilesDir().getAbsolutePath()
            + APP_CACHE_DIRNAME;
    Log.i("cachePath", cacheDirPath);
    // 设置数据库缓存路径
    mWebView.getSettings().setDatabasePath(cacheDirPath); // API 19 deprecated
    // 设置Application caches缓存目录
    mWebView.getSettings().setAppCachePath(cacheDirPath);
    // 开启Application Cache功能
    mWebView.getSettings().setAppCacheEnabled(true);

    Log.i("databasepath", mWebView.getSettings().getDatabasePath());

}

public void clearWebViewCache() {
    // 清理WebView缓存数据库
    try {
        deleteDatabase("webview.db");
        deleteDatabase("webviewCache.db");
    } catch (Exception e) {
        e.printStackTrace();
    }

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

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

    // 删除webView缓存目录
    if (webviewCacheDir.exists()) {
        deleteFile(webviewCacheDir);
    }
    // 删除webView缓存,缓存目录
    if (appCacheDir.exists()) {
        deleteFile(appCacheDir);
    }
}

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

/**
 * 用于控制页面导航
 *
 * @author wwj_748
 *
 */
private class MyWebViewClient extends WebViewClient {
    /**
     * 当用于点击链接,系统调用这个方法
     */
    @Override
    public boolean shouldOverrideUrlLoading(WebView view, String url) {
        if (Uri.parse(url).getHost().equals("www.baidu.com")) {
            // 这个是我的网页,所以不要覆盖,让我的WebView来加载页面
            return false;
        }
        // 否则,这个链接不是我的网站页面,因此启用浏览器来处理urls
        Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
        startActivity(intent);
        return true;
    }
}

@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    // 检查是否为返回事件,如果有网页历史记录
    if (keyCode == KeyEvent.KEYCODE_BACK && mWebView.canGoBack()) {
        mWebView.goBack();
        return true;
    }
    // 如果不是返回键或没有网页浏览历史,保持默认
    // 系统行为(可能会退出该活动)
    return super.onKeyDown(keyCode, event);
}

@Override
public void onClick(View v) {
    switch (v.getId()) {
    case R.id.btn_nightmode:
        mWebView.loadUrl("javascript:load_night()");
        break;
    case R.id.btn_lightmode:
        mWebView.loadUrl("javascript:load_day()");
        break;
    default:
        break;
    }
}

}

转:http://blog.csdn.net/wwj_748/article/details/44835865

时间: 2024-10-25 22:20:24

Android之 -WebView实现离线缓存阅读的相关文章

Android记录25-WebView实现离线缓存阅读

Android记录25-WebView实现离线缓存阅读 前言 本篇博客要实现的是一个离线下载和离线阅读的功能,这是很多阅读类app都常见的一个功能,典型的应用就是网易新闻.什么是离线下载?其实这个概念是比较模糊,是离线之后下载呢,还是下载之后离线,但稍微有点脑子的人都知道没有网络之后怎么下载呢?所以离线下载这个功能是"在有网络的情况下,把资源下载到本地",离线阅读就是"在没有网络或者网络不好的时候,阅读本地好缓存的文章资源".这样就很清楚我们要的这两个具体的功能需求

android中如何实现离线缓存

离线缓存就是在网络畅通的情况下将从服务器收到的数据保存到本地,当网络断开之后直接读取本地文件中的数据. 将网络数据保存到本地: 你可以自己写一个保存数据成本地文件的方法,保存在android系统的任意目录(当然是有权限的才行),但是在这种情况下使用Context的openFileOutput方法最简便也最符合我们的场景,下面的saveObject方法演示了如何用openFileOutput将数据保存在本地的一个文件中: saveObject public static boolean saveO

【转】android中如何实现离线缓存

原文地址:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1209/2136.html 离线缓存就是在网络畅通的情况下将从服务器收到的数据保存到本地,当网络断开之后直接读取本地文件中的数据. 将网络数据保存到本地: 你可以自己写一个保存数据成本地文件的方法,保存在android系统的任意目录(当然是有权限的才行),但是在这种情况下使用Context的openFileOutput方法最简便也最符合我们的场景,下面的saveObj

我的项目10:Android的webview支持HTML5的离线应用功能详细配置

因为需要,我要在andriod手机上用webview给我的HTML5项目加一个壳,但第一次将项目封装进andriod里面时,我测试输出的alert尽然是不支持HTML5 web缓存,这让我很郁闷,在网上找了了一些资料看了一下,测试了几次终于通了,因为我不是搞andriod的,所以就不详细介绍了,把代码分享给大家: MainActivity.java类: package com.example.test; import android.app.Activity; import android.co

关于离线缓存webView的新方法NSURLProtocol

博文转发自:http://blog.sina.com.cn/s/blog_6291e42d0102v0hf.html 对于目前UIWebView的离线缓存方式主要有如下几种: 1.HTML5 , Manifest最开始我的想法是使用HTML5中的离线存储功能,也就是分析Manifest文件来存储和更新部分资源文件.但是经过实践发现,UIWebView根本不支持HTML5,他只实现了Webkit中页面渲染的那一部分.所以要实现缓存必须要另辟蹊径. 2.NSURLCache  尽管在官方的说明文档里

Android 离线缓存的快速实现

离线缓存是指在有网络的状态下将从服务器获取的网络数据,如Json 数据缓存到本地,在断网的状态下启动APP时读取本地缓存数据显示在界面上,常用的APP(网易新闻.知乎等等)都是支持离线缓存的,这样带来了更好的用户体验. 如果能够在调用网络接口后自动缓存返回的Json数据,下次在断网状态下调用这个接口获取到缓存的Json数据的话,那该多好呢?Volley做到了这一点. 因此,今天这篇文章介绍的就是使用Volley自带的数据缓存,配合Universal-ImageLoader的图片缓存,实现断网状态

Android 高级UI设计笔记24:Android 夜间模式之 WebView 实现白天 / 夜间阅读模式 (使用JavaScript)

1. 问题引入: 前面我们是使用方法 降低屏幕亮度(不常用) 和 替换theme,两者都是针对Activity的背景进行白天.夜间模式的交换,但是如果我们显示的是Html的内容,这个时候改怎么办? 分析:首先加载Html页面肯定是要用到WebView的,通过loadUrl的方法可以把html页面显示到webView,我们知道Android可以与JavaScript进行交互,就是说可以在JavaScript中调用Android的代码,也可以在Android中调用JavaScript代码. 所以就有

【Android】WebView读取本地图片

背景 咱的博客园APP,是通过一个WebView来展示新闻的详情的.新闻必然是图文并茂的,无论是支持离线缓存还是加速新闻的打开速度, 都需要咱们打通本地存储与WebView之间的桥梁. 思路 1:首先把WebView的绝对路径设置为我们图片存储的根目录 修改第一个参数以指向本地存储目录,这样就可以使用相对路径引用该目录下的本地文件了.如 webView.loadDataWithBaseURL("file:///storage/emulated/0/Android/data/zhexian.lea

【转】Retrofit 源码解读之离线缓存策略的实现

Retrofit 源码解读之离线缓存策略的实现 Retrofit 是square公司开发的一款网络框架,也是至今Android网络请求中最火的一个,配合OkHttp+RxJava+Retrofit三剑客更是如鱼得水,公司项目重构时,我也在第一时间使用了RxJava+Retrofit,使用过程中遇到的一些问题,也会在后续的博客中,一点点分享出来,供大家参考! 在项目的过程中,项目需求需要在离线的情况下能够继续浏览app内容,第一时间想到缓存,于是经过各种google搜索,得出以下结论(使用Retr