WebView爬坑

1. “Uncaught TypeError: Cannot read property 'getItem' of null”

常见现象:
Webview 加载一些链接出现白板现象,并且Log中打印出上述信息[具体描述]。

解决办法:

WebSettings settings = webView.getSettings();
settings.setDomStorageEnabled(true);

2. "A WebView method was called on thread xxxxxx,All WebView methods must be called on the same thread."

最初工程的TargetAapi 是14,后来改成22之后,出现了上述问题,然而并没有更改任何逻辑代码。
原因:WebView的方法,只能在同一个线程中操作,即只能在主UI线程,即使是调用getUrl这样的方法。
解决办法:将WebView的操作均放在主UI线程执行。

Js交互-H5回调App的方法,其执行所在的线程为非UI线程

3. Alert无法弹出

网页调用altert进行弹窗,app无反应,通常是没有设置WebChromeClient,所以解决办法就是跟webview设置WebChromeClient即可。

网上流传一种WebView的使用方式,就是在界面中new WebView,然后再添加到布局中使用,这个时候如果传给webview的context为application context,哪么即使设置了WebChromeClient也无法弹窗

4.获取网页JS方法的返回值

通常我门有app与网页通过js交互的过程,有些时候我门需要获取网页方法的返回值,这应该是一个非常常见的交互过程。

  • 在4.4以前,获取js的返回值,我门通常是在app中注册一个方法接收返回值,当App调用网页的方法,网页执行完成之后再调用app的方法传递返回值,其流程如下(假如网页有一个getValue方法,返回一个字符串):
  1. app注册js交互方法

   //定义交互的方法
   class JSInterface {
       //由网页调用,获取网页传递的结果
       @JavascriptInterface
       public void onJsResult(int result) {
           Log.i(LOGTAG, "onSumResult result=" + result);
       }
   }
//注册交互方法
mWebView.addJavascriptInterface(new JSInterface(), "jsbridge");
  1. app调用网页方法
String call = "javascript: getValue()";
webView.loadUrl(call);
  1. 网页在触发的方法中回调app返回结果
function getValue(){
       window.jsbridge.onJsResult(1)
}
  • 在4.4之后可以直接使用网页方法的返回值:
 webView.evaluateJavascript("getValue()", new ValueCallback<String>() {

  @Override
  public void onReceiveValue(String value) {
      Log.i(LOGTAG, "onReceiveValue value=" + value);
  }});
}

上面限定了结果返回结果为String,对于简单的类型会尝试转换成字符串返回,对于复杂的数据类型,建议以字符串形式的json返回。
evaluateJavascript方法必须在UI线程(webview的方法调用都应该在UI线程执行,见上述问题2)调用,因此onReceiveValue也执行在主线程

5.Uncaught TypeError: Object [object Object] has no method

原文:

Caution: If you’ve set your targetSdkVersion to 17 or higher, you must add the @JavascriptInterface annotation to any method that you want available your web page code (the method must also be public). If you do not provide the annotation, then the method will not accessible by your web page when running on Android 4.2 or higher.

也就是说如果工程的targetSdkVersion是17及以上,哪么暴露给网页调用的方法,必须增加@JavascriptInterface注解,并且这个方法也必须是public的,如果没有这么操作的话,哪么在4.2及以上的设备上就无法被网页调用

6. 关于混淆

# 混淆注意事项第四条,保持WebView中JavaScript调用的方法
# 下面新增的-keepattributes *Annotation*保留注解,还应该是包含了这句功能
-keepclassmembers class * {
    @android.webkit.JavascriptInterface <methods>;
}
# 建议:使用了WebView和JS的需要添加下面语句
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
-keepclassmembers class fqcn.of.javascript.interface.for.webview {
   public *;
}

更多混淆相关内容,请见我的另一篇文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 175,066评论 25 709
  • 不知不觉,Hybird App已经成了目前比较主流的一种开发方式。 对于用户体验要求较高或者与硬件交互较多的功能我...
    香辣牛肉面阅读 13,522评论 3 89
  • 这篇博客主要来介绍 WebView 的相关使用方法,常见的几个漏洞,开发中可能遇到的坑和最后解决相应漏洞的源码,以...
    Shawn_Dut阅读 12,052评论 3 55
  • WebView·开车指南 目录 WebView简介 WebView基本使用 WebView常用方法 WebSett...
    南城的人阅读 10,184评论 0 19
  • “我跪在故乡的土地上,敬仰地磕着头……”“对这片土地,我是爱得如此深沉……”不少诗人都激情澎湃地歌颂自己的故乡。 ...
    落雁南归阅读 1,345评论 0 2