日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android WebView常见问题及解决方案汇总

發布時間:2024/4/17 Android 52 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android WebView常见问题及解决方案汇总 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

轉載,請聲明出處: 時之沙:?http://blog.csdn.net/t12x3456


Android WebView常見問題解決方案匯總:

就目前而言,如何應對版本的頻繁更新呢,又如何靈活多變地展示我們的界面呢,這又涉及到了web app與native app之間孰優孰劣的爭論. 于是乎,一種混合型的app誕生了,靈活多變的部分,如淘寶商城首頁的活動頁面,一集凡客誠品中我們都可以見到web 頁面與native頁面的混合,既利用了web app的靈活易更新,也借助了native app本身的效率.
當然,就會用到webview這樣的一個控件,這里,我把自己使用過程中遇到的一些問題整理下來.

首先上張圖對WebView進行一個基本的回顧:



以上思維導圖原文件下載地址:

http://download.csdn.net/detail/t12x3456/6509195


然后看一下具體的問題及解決方案:

1.為WebView自定義錯誤顯示界面:

覆寫WebViewClient中的onReceivedError()方法:

[java]?view plaincopy
  • /**?
  • ?*?顯示自定義錯誤提示頁面,用一個View覆蓋在WebView?
  • ?*/??
  • protected?void?showErrorPage()?{??
  • ????LinearLayout?webParentView?=?(LinearLayout)mWebView.getParent();??
  • ??????
  • ????initErrorPage();??
  • ????while?(webParentView.getChildCount()?>?1)?{??
  • ????????webParentView.removeViewAt(0);??
  • ????}??
  • ????LinearLayout.LayoutParams?lp?=?new?LinearLayout.LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT);??
  • ????webParentView.addView(mErrorView,?0,?lp);??
  • ????mIsErrorPage?=?true;??
  • }??
  • protected?void?hideErrorPage()?{??
  • ????LinearLayout?webParentView?=?(LinearLayout)mWebView.getParent();??
  • ??????
  • ????mIsErrorPage?=?false;??
  • ????while?(webParentView.getChildCount()?>?1)?{??
  • ????????webParentView.removeViewAt(0);??
  • ????}??
  • }??
  • ??
  • ??
  • ???protected?void?initErrorPage()?{??
  • ????if?(mErrorView?==?null)?{??
  • ????????mErrorView?=?View.inflate(this,?R.layout.online_error,?null);??
  • ????????Button?button?=?(Button)mErrorView.findViewById(R.id.online_error_btn_retry);??
  • ????????button.setOnClickListener(new?OnClickListener()?{??
  • ????????????public?void?onClick(View?v)?{??
  • ????????????????mWebView.reload();??
  • ????????????}??
  • ????????});??
  • ????????mErrorView.setOnClickListener(null);??
  • ????}??
  • }??
  • [java]?view plaincopy
  • ??
  • [java]?view plaincopy
  • ??
  • [java]?view plaincopy
  • @Override??
  • public?void?onReceivedError(WebView?view,?int?errorCode,?String?description,?String?failingUrl)?{??
  • <span?style="white-space:pre">??????????????????</span>??
  • <span?style="white-space:pre">??????????</span>mErrorView.setVisibility(View.VISIBLE);??
  • <span?style="white-space:pre">??????????</span>super.onReceivedError(view,?errorCode,?description,?failingUrl);??
  • }??????????


  • 2.WebView cookies清理:

    [java]?view plaincopy
  • CookieSyncManager.createInstance(this);???
  • CookieSyncManager.getInstance().startSync();???
  • CookieManager.getInstance().removeSessionCookie();???

  • 3.清理cache 和歷史記錄:

    [java]?view plaincopy
  • webView.clearCache(true);???
  • webView.clearHistory();??

  • 4.判斷WebView是否已經滾動到頁面底端:

    [java]?view plaincopy
  • getScrollY()方法返回的是當前可見區域的頂端距整個頁面頂端的距離,也就是當前內容滾動的距離.???
  • getHeight()或者getBottom()方法都返回當前WebView?這個容器的高度???
  • getContentHeight?返回的是整個html?的高度,但并不等同于當前整個頁面的高度,因為WebView?有縮放功能,?所以當前整個頁面的高度實際上應該是原始html?的高度再乘上縮放比例.?因此,更正后的結果,準確的判斷方法應該是:???
  • if(WebView.getContentHeight*WebView.getScale()?==?(webview.getHeight()+WebView.getScrollY())){?//已經處于底端?}???

  • 5.URL攔截:

    Android WebView是攔截不到頁面內的fragment跳轉的。但是url跳轉的話,又會引起頁面刷新,H5頁面的體驗又下降了。只能給WebView注入JS方法了。



    ?6.處理WebView中的非超鏈接請求(如Ajax請求):?

    ?有時候需要加上請求頭,但是非超鏈接的請求,沒有辦法再shouldOverrinding中攔截并用webView.loadUrl(String url,HashMap headers)方法添加請求頭

    ? 目前用了一個臨時的辦法解決:

    首先需要在url中加特殊標記/協議, 如在onWebViewResource方法中攔截對應的請求,然后將要添加的請求頭,以get形式拼接到url末尾

    在shouldInterceptRequest()方法中,可以攔截到所有的網頁中資源請求,比如加載JS,圖片以及Ajax請求等等

    Ex:

    [java]?view plaincopy
  • @SuppressLint("NewApi")??
  • @Override??
  • public?WebResourceResponse?shouldInterceptRequest(WebView?view,String?url)?{??
  • ????//?非超鏈接(如Ajax)請求無法直接添加請求頭,現拼接到url末尾,這里拼接一個imei作為示例??
  • ??
  • ????String?ajaxUrl?=?url;??
  • ????//?如標識:req=ajax??
  • ????if?(url.contains("req=ajax"))?{??
  • ???????ajaxUrl?+=?"&imei="?+?imei;??
  • ????}??
  • ??
  • ????return?super.shouldInterceptRequest(view,?ajaxUrl);??
  • ??
  • }??



  • 7.在頁面中先顯示圖片:

    [java]?view plaincopy
  • @Override??
  • public?void?onLoadResource(WebView?view,?String?url)?{??
  • ??mEventListener.onWebViewEvent(CustomWebView.this,?OnWebViewEventListener.EVENT_ON_LOAD_RESOURCE,?url);??
  • ????if?(url.indexOf(".jpg")?>?0)?{??
  • ?????hideProgress();?//請求圖片時即顯示頁面??
  • ?????mEventListener.onWebViewEvent(CustomWebView.this,?OnWebViewEventListener.EVENT_ON_HIDE_PROGRESS,?view.getUrl());??
  • ?????}??
  • ????super.onLoadResource(view,?url);??
  • }??


  • ?8.屏蔽掉長按事件 因為webview長按時將會調用系統的復制控件:
    ?? ???

    [java]?view plaincopy
  • mWebView.setOnLongClickListener(new?OnLongClickListener()?{??
  • ????????????
  • ??????????@Override??
  • ??????????public?boolean?onLongClick(View?v)?{??
  • ??????????????return?true;??
  • ??????????}??
  • ??????});??

  • 9.在WebView加入 flash支持:

    [java]?view plaincopy
  • String?temp?=?"<html><body?bgcolor=\""?+?"black"??
  • ????????????????+?"\">?<br/><embed?src=\""?+?url?+?"\"?width=\""?+?"100%"??
  • ????????????????+?"\"?height=\""?+?"90%"?+?"\"?scale=\""?+?"noscale"??
  • ????????????????+?"\"?type=\""?+?"application/x-shockwave-flash"??
  • ????????????????+?"\">?</embed></body></html>";??
  • String?mimeType?=?"text/html";??
  • String?encoding?=?"utf-8";??
  • web.loadDataWithBaseURL("null",?temp,?mimeType,?encoding,?"");??

  • 10.WebView保留縮放功能但隱藏縮放控件:

    [java]?view plaincopy
  • mWebView.getSettings().setSupportZoom(true);??
  • ????????mWebView.getSettings().setBuiltInZoomControls(true);??
  • ????????if?(DeviceUtils.hasHoneycomb())??
  • ??????????????mWebView.getSettings().setDisplayZoomControls(false);??

  • 注意:setDisplayZoomControls是在Android 3.0中新增的API.


    這些是目前我整理出來的一些注意事項和問題解決方案,也歡迎大家多提一些關于webview的問題,如果有合適的解決方案,我會直接更新到這篇文章.


    8月份更新:

    11.WebView 在Android4.4的手機上onPageFinished()回調會多調用一次(具體原因待追查)

    需要盡量避免在onPageFinished()中做業務操作,否則會導致重復調用,還有可能會引起邏輯上的錯誤.


    12.需要通過獲取Web頁中的title用來設置自己界面中的title及相關問題:

    需要給WebView設置 WebChromeClient,并在onReceiveTitle()回調中獲取

    [java]?view plaincopy
  • WebChromeClient?webChromeClient?=?new?WebChromeClient()?{????
  • ????????????@Override????
  • ????????????public?void?onReceivedTitle(WebView?view,?String?title)?{????
  • ????????????????super.onReceivedTitle(view,?title);????
  • ?????????????????
  • ????????????????txtTitle.setText(title);????
  • ????????????}????
  • ????
  • ????????};????

  • 但是發現在小米3的手機上,當通過webview.goBack()回退的時候,并沒有觸發onReceiveTitle(),這樣會導致標題仍然是之前子頁面的標題,沒有切換回來.

    這里可以分兩種情況去處理:

    (1) 可以確定webview中子頁面只有二級頁面,沒有更深的層次,這里只需要判斷當前頁面是否為初始的主頁面,可以goBack的話,只要將標題設置回來即可.

    (2)webview中可能有多級頁面或者以后可能增加多級頁面,這種情況處理起來要復雜一些:

    ??? 因為正常順序加載的情況onReceiveTitle是一定會觸發的,所以就需要自己來維護webview? loading的一個url棧及url與title的映射關系

    ?? 那么就需要一個ArrayList來保持加載過的url,一個HashMap保存url及對應的title.

    ?? 正常順序加載時,將url和對應的title保存起來,webview回退時,移除當前url并取出將要回退到的web 頁的url,找到對應的title進行設置即可.


    這里還要說一點,當加載出錯的時候,比如無網絡,這時onReceiveTitle中獲取的標題為 找不到該網頁,因此建議當觸發onReceiveError時,不要使用獲取到的title.


    13.WebView因addJavaScriptInterface()引起的安全問題.

    ??? 這個問題主要是因為會有惡意的js代碼注入,尤其是在已經獲取root權限的手機上,一些惡意程序可能會利用該漏洞安裝或者卸載應用.

    關于詳細的情況可以參考下面這篇文章:

    .http://blog.csdn.net/leehong2005/article/details/11808557

    ? ?

    還有一個開源項目可以參考:?https://github.com/pedant/safe-java-js-webview-bridge, 該項目利用onJsPrompt() 替代了addJavaScriptInterface(),(解決方案類似上述參考的博客)同時增加了異步回調,

    很好地解決了webview ?js注入的安全問題.


    10月份更新:

    14.WebView頁面中播放了音頻,退出Activity后音頻仍然在播放

    需要在Activity的onDestory()中調用

    [java]?view plaincopy
  • webView.destroy();??

  • 但是直接調用可能會引起如下錯誤:

    [java]?view plaincopy
  • 10-10?15:01:11.402:?E/ViewRootImpl(7502):?sendUserActionEvent()?mView?==?null??
  • 10-10?15:01:26.818:?E/webview(7502):?java.lang.Throwable:?Error:?WebView.destroy()?called?while?still?attached!??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.webkit.WebViewClassic.destroy(WebViewClassic.java:4142)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.webkit.WebView.destroy(WebView.java:707)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?com.didi.taxi.ui.webview.OperatingWebViewActivity.onDestroy(OperatingWebViewActivity.java:236)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.Activity.performDestroy(Activity.java:5543)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.Instrumentation.callActivityOnDestroy(Instrumentation.java:1134)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.performDestroyActivity(ActivityThread.java:3619)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:3654)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.access$1300(ActivityThread.java:159)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.ActivityThread$H.handleMessage(ActivityThread.java:1369)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.os.Handler.dispatchMessage(Handler.java:99)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.os.Looper.loop(Looper.java:137)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.main(ActivityThread.java:5419)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?java.lang.reflect.Method.invokeNative(Native?Method)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?java.lang.reflect.Method.invoke(Method.java:525)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1187)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003)??
  • 10-10?15:01:26.818:?E/webview(7502):????at?dalvik.system.NativeStart.main(Native?Method)??

  • 如上所示,webview調用destory時,webview仍綁定在Activity上.這是由于自定義webview構建時傳入了該Activity的context對象,因此需要先從父容器中移除webview,然后再銷毀webview:

    [java]?view plaincopy
  • rootLayout.removeView(webView);??
  • webView.destroy();??

  • 15. WebView長按自定義菜單,實現復制分享相關功能

    ?這個功能首先可以從兩方面完成:

    ? (1) 在js中完成:

    ? ? 處理android.selection.longTouch

    這里推薦一個開源項目進行參考,:

    https://github.com/btate/BTAndroidWebViewSelection

    ???(2) 安卓層處理:

    ? ? ?首先使用OnTouchListener實現長按實現監聽,然后實現WebView的Context menu,最后調用webview中的emulateShiftHeld(),為了適配安卓不同版本,最好使用反射方式調用.


    總結

    以上是生活随笔為你收集整理的Android WebView常见问题及解决方案汇总的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。