webapp开发webview缓存问题

目前webapp越来越多,体验也越来越好,为了能够更好的使用WebView展示出流畅的的页面,可以从以下几点做优化:

  • WebView缓存
  • 资源文件本地存储
  • 减少耗时操作
  • 客户端UI优化

可能有人会说了,为什么不做成native的呢,这样就不用那么的麻烦了。如果我需要加载的内容都是静态的,当然做成native的是最好的,为什么我们要使用WebView呢,因为它可以加载一些容易改变的内容,同时也方便制作多平台应用。

WebView可以优化的哪些地方呢?

WebView缓存

开启WebView的缓存功能可以减少对服务器资源的请求,一般使用默认缓存策略就可以了。

//设置 缓存模式 
webView.getSettings().setCacheMode(WebSettings.LOAD_DEFAULT);  
// 开启 DOM storage API 功能 
webView.getSettings().setDomStorageEnabled(true); 

资源文件本地存储

资源等文件(不需要更新)本地存储,在需要的时候直接从本地获取。哪些资源需要我们去存储在本地呢,当然是一些不会被更新的资源,例如图片文件,js文件,css文件,替换的方法也很简单,重写WebView的方法即可。

{
      try {
	      if (url.endsWith("icon.png")) {
	          InputStream is = appRm.getInputStream(R.drawable.icon);
	          WebResourceResponse response = new WebResourceResponse("image/png",
	            "utf-8", is);
	          return response;
	      } else if (url.endsWith("jquery.min.js")) {
	          InputStream is = appRm.getInputStream(R.raw.jquery_min_js);
	          WebResourceResponse response = new WebResourceResponse("text/javascript",
	            "utf-8", is);
	          return response;
	      }
	      } catch (IOException e) {
	     		 e.printStackTrace();
	      }
	      return super.shouldInterceptRequest(view, url);
}
  1. appRm为app资源管理器,读取drawable,assets,raw下的资源,都是Android系统的一些很简单的函数调用。

  2. getInputStream的参数代表资源具体位置

  3. WebResourceResponse后的资源类型需要写正确

有些时候我们会为我们的网站加入一些统计代码,这些也可以精简掉(自己使用的CNZZ的大概占的有10k左右),可以使用Charles对客户端进行抓包查看。

减少耗时操作

准确的说,是减少同步操作的操作时间,尽量使用异步操作替代同步操作。如果服务端存在读取数据库和计算耗时的操作,尽量使用异步(ajax)进行操作,把原本的时间花在异步操作上。

举个例子,A页面到B页面,A页面实现登录功能,B页面展示主功能页面,如果让B页面去进行用户登录信息验证的话,B页面加载时间会加长(数据库查询等操作),同时客户端可能需要提供一个等待框(或进度条等)给用户,那看看在A页面使用异步操作的优势吧:

  • 可以提供统一的js等待框,多平台保持一致性,减少客户端代码工作量。
  • 加载页面的时间变短。B页面由于减少了耗时的操作,加载时间变短,用户等待时间也变短。
  • 可以方便加入一些验证后的控制逻辑,不需要进行页面跳转。A页面可以根据异步操作进行结果判断,做出相应的处理。

客户端UI优化

怎么让用户看不到WebView加载前的白色页面呢?首次加载后页面的跳转可以用上面的步骤进行优化,可以提供给用户一个很好的体验,那加载的第一页呢?我们需要WebView预加载页面,这个该怎么做到的呢?下面提供两种方法:

  • ViewPager,将欢迎页面与WebView页面一起放进ViewPager中,设置预加载页面个数,使WebView所在页面可以预加载,在加载完毕的时候切换到WebView所在页面。
  • FrameLayout,将欢迎页面与WebView页面的布局合在一起,显示在一个页面内,起始隐藏WebView布局,待WebView加载完毕,隐藏欢迎布局,显示WebView布局。

使用FrameLayout简单一些,两种方法都是需要对WebChromeClient的onProgressChanged进行监听,加载完毕进行页面切换,如下:

  webView.setWebChromeClient(new WebChromeClient() {
        @Override
        public void onProgressChanged(WebView view, int newProgress) {
            super.onProgressChanged(view, newProgress);
            if (newProgress >= 100) {
                // 切换页面
            }
        }
    });

经过以上几步的优化,一个流畅的webapp生成了。

贴代码


  
import java.io.File;  
  




import com.example.wanmengapp.R;


import Util.ShowDialog;
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 WebViewActivity extends Activity {  
  
    private static final String TAG = WebViewActivity.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.webview);  
          
        //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.text_title);  
          
      //  rl_loading = (RelativeLayout) findViewById(R.id.);  
          
        mWebView = (WebView) findViewById(R.id.webView);  
          
        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");  
                  ShowDialog.showmeidialog(WebViewActivity.this, "加载中...");
            //    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);  
  ShowDialog.dimissdialog();
            //    rl_loading.setVisibility(View.GONE); // 隐藏加载界面  
            }  
  
            @Override  
            public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {  
             ShowDialog.dimissdialog();
              //  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());  
        }  
    }  
  
}  

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值