21

我知道要从 Javascript 交互到 Java,您必须使用 webview 中的 addjavascriptInterface 方法注入 Java 对象。

这是我面临的问题。

  1. 我使用方法注册了一个 java 对象addJavascriptInterface,以便在我的 JS 中可用。

  2. 我使用在 webview 中注入了一些 JSwebview.loadURL("javascript:XXX");

  3. 当我完成注入 JS 时,我会发送一个 JS 事件。

问题是,如果在第 1 步之后立即执行以下 Javascript:

mWebView.loadUrl("javascript:if(window.myobject) console.log('myobject found---------'); else {console.log('myobject not found----');}");

我在控制台的日志中收到“找不到我的对象”。

我想知道,如果有一段时间我可以访问我的对象,如果是这样,我如何知道我应该等待多长时间才能调用我的对象?

4

3 回答 3

48

我想知道是否有一段时间我可以访问我的对象

是的,我认为有延迟,因为WebView.addJavascriptInterface会在 WebView 的内部工作线程中运行。或许你已经想到了这一点,意识到 WebView 至少要维护一个工作线程来做异步网络 IO。也许您在使用 WebView 时也注意到了 DDMS 中的这些线程。

事实证明,它还使用一个线程来为许多其他公共方法工作。我真的希望 Google 的文档能更清楚地说明这一点!但我希望我能帮助你,并向你展示我是如何为自己确认这一点的。

跟我来看看 WebView 的源代码。它具有合理的可读性,即使您无法准确了解正在发生的事情,也可以通过回答有关线程的一些问题来追踪。

您可以通过 SDK 管理器工具下载 Android 框架源,但它也镜像在 Github 上,所以我在这里链接到了。我猜测并选择了一个接近 ICS 版本的标签。不难找到WebView.addJavascriptInterface。我刚刚用谷歌搜索了“WebView.java 站点:github.com/android”。

该方法WebView.addJavascriptInterface向以下实例发送消息WebViewCore

mWebViewCore.sendMessage(EventHub.ADD_JS_INTERFACE, arg);

里面有WebViewCore.java一堆重载的方法被称为sendMessage,但我们并不需要知道究竟是哪一个被调用,因为它们做的事情几乎相同。甚至有一个很好的评论给我们一个提示,我们在正确的地方!所有这些都委托给一个实例,EventHub该实例是某个内部类。这个方法结果是同步的,并且正在向 的实例发送消息Handler,这很好地表明它可能正在另一个线程中运行,但为了完整起见,让我们找出来!

那是从 调用Handler的实例化的。这里还有几跳,但最终我发现这是从(的子类)中调用的,它与新的右在这里一起实例化。EventHub.transferMessagesWebViewCore.initializerunWebCoreThreadRunnableThread

多么冒险!所以,即使我真的不能确定所有这些移动部件发生了什么,我很有信心说这种方法不是同步的,而是向 WebView 的工作线程发送消息。我希望这是有道理的!

如果是这样,我如何知道我应该等待多长时间才能调用我的对象?

不幸的是,我不知道这个问题的答案。我正在研究这个确切的问题,并在我的谷歌搜索过程中在 StackOverflow 上发现了这个问题。我认为您有以下选择,其中一些比其他更好或更容易:

1) 仅Thread.sleep100 毫秒或介于addJavascriptInterface和之间的时间loadUrl("javascript:...")。Blech,我不喜欢这个,但它可能是最简单的。

2) 另一种可能性是,您可以WebView.loadUrl使用专门测试接口是否已设置的 JavaScript 片段进行调用,并捕获未设置时引发的 ReferenceError。但是,正如您可能已经猜到的那样,这种方法涉及向 WebView 添加一个 JavaScript 接口!

3) 改为调用WebView.setWebChromeClient,并捕获 JavaScript 的alert()or console.log。从我的实验来看,这种方法同步的,所以没有延迟。(我已经在源代码中确认了这一点,但我将把细节作为练习留给读者)你可能应该想出一些特殊的字符串来调用alert并在里面检查它onJsAlert,所以你不只是捕捉所有alert()的 s。

对不起这个答案的长度,我希望这会有所帮助。祝你好运!

于 2012-07-20T04:33:03.087 回答
1

确保您需要从 Java 访问的 HTML / Javascript 中声明的 Javascript 对象声明为全局的,否则它们很可能会被收集。我有执行此操作的代码(其中 Android 是我使用 addJavascriptInterface 添加的界面):

<script>
  var cb = function(location) {
     alert('location is ' + location);
  }
  Android.getLocation('cb');
</script>

getLocation 方法调用 Android 的 LocationManager.requestSingleUpdate,然后在 LocationListener 触发时调用回调。

如果没有“var”,我发现当位置查找调用回调时,回调函数已被垃圾收集。

于 2013-03-05T05:11:44.523 回答
0

(从我对类似问题的回答中复制)

我将 Jason Shah 和 Mr S 的实施作为我修复的基石,并对其进行了极大的改进。

有太多代码可以放入此评论中,我将链接到它。

要点是:

  • 适用于所有版本的 Gingerbread (2.3.x)
  • 从 JS 到 Android 的调用现在是同步的
  • 不再需要手动映射接口方法
  • 修复了字符串分隔符破坏代码的可能性
  • 更改 JS 签名和接口名称要容易得多
于 2013-09-17T00:33:45.073 回答