android webview 使用以及一些常见的异常处理

先来看个实例:

 

  1. public class MainActivity extends Activity {
  2.     final String COMPANY_WEB=”http://www.csdn.net”;
  3.     private WebView mWebView;
  4. [email protected]
  5.     protected void onCreate(Bundle savedInstanceState) {
  6.         super.onCreate(savedInstanceState);
  7.         setContentView(R.layout.activity_main);
  8.         mWebView = (WebView) findViewById(R.id.webview);
  9.         setWebView();
  10.     }
  11.     private void setWebView(){
  12.         WebSettings webSettings = mWebView.getSettings();
  13.           webSettings.setJavaScriptEnabled(true);
  14.           webSettings.setAllowFileAccess(true);
  15.           webSettings.setBuiltInZoomControls(true);
  16.           webSettings.setPluginsEnabled(true);
  17.         mWebView.setWebViewClient(new MonitorWebClient());
  18.         mWebView.loadUrl(COMPANY_WEB);
  19.     }
  20.     private class MonitorWebClient extends WebViewClient{
  21. [email protected]
  22.         public void onPageStarted(WebView view, String url, Bitmap favicon) {
  23.             super.onPageStarted(view, url, favicon);
  24.         }
  25. [email protected]
  26.         public void onPageFinished(WebView view, String url) {
  27.             super.onPageFinished(view, url);
  28.         }
  29. [email protected]
  30.         public boolean shouldOverrideUrlLoading(WebView view, final String url) {
  31.             String website=Uri.parse(url).getHost();
  32.             if (COMPANY_WEB.equals(website)) {
  33.                     // This is my web site, so do not override; let my WebView load the page
  34.                     return false;
  35.               }else{
  36.                  view.loadUrl(url);
  37.                  return true;
  38.              }
  39.         //  return super.shouldOverrideUrlLoading(view, url);
  40.         }
  41.     }
  42. [email protected]
  43.     public boolean onKeyDown(int keyCode, KeyEvent event)
  44.     {
  45.         if ((keyCode == KeyEvent.KEYCODE_BACK) && mWebView.canGoBack())
  46.         {
  47.             mWebView.goBack();
  48.             return true;
  49.         }
  50.         return super.onKeyDown(keyCode, event);
  51.     }
  52. }

 

 

相关权限:

 

  1. <uses-permission android:name=”android.permission.INTERNET” />
  2.     <uses-permission android:name=”android.permission.ACCESS_NETWORK_STATE” />
  3.     <uses-permission android:name=”android.permission.ACCESS_WIFI_STATE” />

 

 

ok,测试了一下相关链接也可以出现正常访问,以上是以csdn网站为例,如果将网站换成http://www.qq.com开始也没有问题, 点击导航栏也可以正常访问,再点击图片连接就会出现 eventhub.removemessages(int what = 107) is not supported before the webviewcore is set up 异常信息,有人说是没有以http://开头,这个也试了一下,没有解决问题,期待有人能解决。回过头来在首页点击相关新闻链接后会发现出现空白页无法正常访问,后来研究发现这个和网站结构有关系,看来webview并不能完全实现浏览器功能。

接下来就是简单的 异常 处理了,主要就是重写WebViewClient类中的onReceivedError()方法和onReceivedSslError()方法来进行 处理了。

说完 异常简单 处理后再来说说提高网站的访问速度,尤其是带有大量的flash,swf动画和各种css样式功能,这个时候我们就应该 使用缓存了,适当的设置缓存大小 以及合适的模式来进行优化了,webview有两种模式可设置如下:

1,LOAD_DEFAULT,根据cache-control决定是否从 网络上取数据。
2,LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都 使用缓存中的数据。
如:m.taobao.com的cache-control为no-cache,在模式LOAD_DEFAULT下,无论如何都会从 网络上取数据,如果没有 网络,就会出现错误页面;在LOAD_CACHE_ELSE_NETWORK模式下,无论是否有 网络,只要打开过一次,都 使用缓存。
m.sina.com.cn的cache-control为max-age=60,在两种模式下都 使用本地缓存数据。

总结:根据以上两种模式,建议缓存策略为,判断是否有 网络,有的话, 使用LOAD_DEFAULT,无 网络时, 使用LOAD_CACHE_ELSE_NETWORK。
好说的也差不多了,来看一下优化后的代码:

 

  1. public class MainActivity extends Activity {
  2.     final String COMPANY_WEB=”http://www.deczh.com/”;
  3.     private WebView mWebView;
  4.     private Context activity;
  5. //  private ProgressDialog progressDialog;
  6.     //history web site
  7. //  private Stack<String> webHistory=new Stack<String>();
  8. [email protected]
  9.     protected void onCreate(Bundle savedInstanceState) {
  10.         super.onCreate(savedInstanceState);
  11.         setContentView(R.layout.activity_main);
  12.         mWebView = (WebView) findViewById(R.id.webview);
  13.         setWebView();
  14.         activity=this;
  15.         mHandler.sendEmptyMessageDelayed(0, 1000);
  16.     }
  17.     private void setWebView(){
  18.         WebSettings webSettings = mWebView.getSettings();
  19.          //java script
  20.           webSettings.setJavaScriptEnabled(true);
  21.           webSettings.setJavaScriptCanOpenWindowsAutomatically(true);
  22.           // access Assets and resources
  23.           webSettings.setAllowFileAccess(true);
  24.           //zoom page
  25.           webSettings.setBuiltInZoomControls(true);
  26.           webSettings.setPluginsEnabled(true);
  27.           //set xml dom cache
  28.           webSettings.setDomStorageEnabled(true);
  29.           //提高渲染的优先级
  30.           webSettings.setRenderPriority(RenderPriority.HIGH);
  31.           //set cache
  32.           String appCachePath = getDir(“netCache”, Context.MODE_PRIVATE).getAbsolutePath();
  33.           webSettings.setAppCacheEnabled(true);
  34.           webSettings.setAppCachePath(appCachePath);
  35.           webSettings.setAppCacheMaxSize(1024*1024*5);
  36.           webSettings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);
  37.         mWebView.setWebViewClient(new MonitorWebClient());
  38.         mWebView.setWebChromeClient(new AppCacheWebChromeClient());
  39.     }
  40.     private class MonitorWebClient extends WebViewClient{
  41. [email protected]
  42.         public void onReceivedError(WebView view, int errorCode,
  43.                 String description, String failingUrl) {
  44.             //错误提示
  45.             Toast toast = Toast.makeText(getBaseContext(), “Oh no! ” + description,
  46.                     Toast.LENGTH_LONG);
  47.             toast.setGravity(Gravity.TOP | Gravity.CENTER, 0, 0);
  48.             toast.show();
  49.             //错误<strong>处理</strong>
  50.             try {
  51.                 mWebView.stopLoading();
  52.             } catch (Exception e) {
  53.             }
  54.             try {
  55.                 mWebView.clearView();
  56.             } catch (Exception e) {
  57.             }
  58.             if (mWebView.canGoBack()) {
  59.                 mWebView.goBack();
  60.             }
  61.         //  super.onReceivedError(view, errorCode, description, failingUrl);
  62.         }
  63.         //当load有ssl层的https页面时,如果这个网站的安全证书在Android无法得到认证,WebView就会变成一个空白页,而并不会像PC浏览器中那样跳出一个风险提示框
  64. [email protected]
  65.         public void onReceivedSslError(WebView view, SslErrorHandler handler,
  66.                 SslError error) {
  67.             //忽略证书的错误继续Load页面内容
  68.              handler.proceed();
  69.             //handler.cancel(); // Android默认的<strong>处理</strong>方式
  70.              //handleMessage(Message msg); // 进行其他<strong>处理</strong>
  71.         //  super.onReceivedSslError(view, handler, error);
  72.         }
  73. [email protected]
  74.         public void onPageStarted(WebView view, String url, Bitmap favicon) {
  75.              /*if (progressDialog == null) {
  76.                  // If no progress dialog, make one and set message
  77.                  progressDialog = new ProgressDialog(activity);
  78.                  progressDialog.setMessage(“Loading please wait…”);
  79.                  progressDialog.show();
  80.                  // Hide the webview while loading
  81.                  mWebView.setEnabled(false);
  82.              }*/
  83.         //  super.onPageStarted(view, url, favicon);
  84.         }
  85. [email protected]
  86.         public void onPageFinished(WebView view, String url) {
  87.            /* if (progressDialog != null&&progressDialog.isShowing()) {
  88.                 progressDialog.dismiss();
  89.                 progressDialog = null;
  90.                 mWebView.setEnabled(true);
  91.             }*/
  92.             /*if(!webHistory.contains(url))
  93.                 webHistory.push(url);*/
  94.         //  super.onPageFinished(view, url);
  95.         }
  96. [email protected]
  97.         public boolean shouldOverrideUrlLoading(WebView view, final String url) {   Log.e(getClass().getSimpleName(), “website= “+url);
  98.         //  String website=Uri.parse(url).getHost();
  99.             String processUrl=url;
  100.             if(!processUrl.startsWith(“http://”))
  101.                 processUrl=”http://”+processUrl;
  102.             if (COMPANY_WEB.equals(url)) {
  103.                // This is my web site, so do not override; let my WebView load the page
  104.                  return false;
  105.               }
  106.             else{
  107.                  view.loadUrl(processUrl);
  108.                  return true;
  109.              }
  110.         //   return super.shouldOverrideUrlLoading(view, url);
  111.         }
  112.     }
  113.      private class AppCacheWebChromeClient extends WebChromeClient {
  114. [email protected]
  115.             public void onReachedMaxAppCacheSize(long spaceNeeded, long totalUsedQuota, WebStorage.QuotaUpdater quotaUpdater) {
  116.             //    Log.e(APP_CACHE, “onReachedMaxAppCacheSize reached, increasing space: ” + spaceNeeded);
  117.                 quotaUpdater.updateQuota(spaceNeeded * 2);
  118.             }
  119.         }
  120.     private boolean pause=false;
  121. [email protected]
  122.     public void onPause() {
  123.         super.onPause();
  124.         if (mWebView != null) {
  125.             mWebView.pauseTimers();
  126.             mWebView.onPause();
  127.             this.pause=true;
  128.         }
  129.     }
  130. [email protected]
  131.     public void onResume() {
  132.         super.onResume();
  133.         if (mWebView != null&&pause) {
  134.             mWebView.resumeTimers();
  135.             mWebView.onResume();
  136.             this.pause=false;
  137.         }
  138.     }
  139. [email protected]
  140.     public boolean onKeyDown(int keyCode, KeyEvent event)
  141.     {
  142.         if ((keyCode == KeyEvent.KEYCODE_BACK) && mWebView.canGoBack()){
  143.             mWebView.goBack();
  144.             return true;
  145.         }
  146.         return super.onKeyDown(keyCode, event);
  147.     }
  148. }

相关权限:

 

 

  1. <uses-permission android:name=”android.permission.INTERNET” />
  2.     <uses-permission android:name=”android.permission.ACCESS_NETWORK_STATE” />
  3.     <uses-permission android:name=”android.permission.ACCESS_WIFI_STATE” />
  4.     <uses-permission android:name=”android.permission.WRITE_EXTERNAL_STORAGE” />
  5.     <uses-permission android:name=”android.permission.READ_EXTERNAL_STORAGE” />

经过代码优化后,访问速度明显提升,部分错误得到 处理,但是还是会有 异常 处理不了,如果谁有好的错误 处理方法,希望留言讨论,大家一起进步。

Tagged:

Comments are closed.