WebView JavaScript 接口似乎只能通过 Android Studio 的 Instant-Run 工作,但不能通过常规 gradle 编译

Posted

技术标签:

【中文标题】WebView JavaScript 接口似乎只能通过 Android Studio 的 Instant-Run 工作,但不能通过常规 gradle 编译【英文标题】:WebView JavaScript Interface only seems to work through the Android Studio's Instant-Run, but not when compiled through regular gradle 【发布时间】:2017-06-10 02:00:46 【问题描述】:

设置

我有一个托管 WebView 组件的简单片段。在这个片段的初始化过程中,我向我的应用注册了一些 javascript 接口方法:

mJavaScriptHooks = new JavaScriptHooks();
...
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.addJavascriptInterface(mJavaScriptHooks, "android");

我的 JavaScript 钩子并没有什么特别之处,并且都以以下方式声明:

public class JavaScriptHooks 
    @JavascriptInterface
    public void someMethod() 
       ...
    
    ...

为了确保这些方法不会被混淆或优化而被遗忘,我添加了以下我的ProGuard configuration:

-keepattributes JavascriptInterface
-keepclassmembers class * 
    @android.webkit.JavascriptInterface <methods>;

问题

一切正常。网页通过提供的“Android”对象成功调用方法,调用方法并执行相应的任务。 但是,它仅在我使用通过 Android Studio 的 Instant-Run 功能生成的 APK 运行应用程序时才有效。

APK 是通过 Android Studio 安装还是我通过命令行使用 adb install 安装都没关系

如果我通过常规 gradle 编译我的 APK(或者如果我禁用 Instant-Run),那么 JavaScript 界面将停止工作。

调试

首先,我确保 ProGuard禁用。即使 ProGuard 以某种方式 ✨ 神奇地 ✨ 启用,我的 ProGuard 配置也应该确保我的 JS 方法不受影响。

我还对 APK 进行了反汇编,以确保通过 JS 接口公开的方法没有以某种方式被破坏(或删除)。他们不是。正如声明的那样,它们似乎存在于 DEX 文件中。

当我运行启用 Instant-Run 生成的 APK 时,我可以(通过 Chrome)检查 WebView 中加载的页面并查看我的“Android”对象是否可用,并且我公开的所有方法都是那里(和可调用的):

> Android
Object
    someMethod()
    __proto__

但是,当我在禁用 Instant-Run(即 vanilla gradle)的情况下运行 APK 时,“Android”对象可用,但我的方法都不存在。它只是一个空对象。

> Android
Object
    __proto__

我在 WebView 中加载的页面位于我控制的公开可见的服务器上。在两个 APK 构建中加载相同的页面,并且以相同的方式调用方法。似乎只有使用 Instant-Run 功能构建的 APK 才有效。

我整天都在挖掘,但我找不到任何真正有用的东西。大多数建议都指向 ProGuard 配置问题或 WebView 中的旧错误(即 Gingerbread 时代)。

【问题讨论】:

你的最小 SDK 是多少? 如何禁用 proguard?在 build.gradle 中使用 minifyEnabled false? @NageshSusarla,最低 SDK 版本为 19 (4.4)。是的,minifyEnabled 是错误的。 ProGuard 在 buildTypes 下被禁用。我没有为调试类型设置 proguardFiles(这是我的目标)。 @MDV Dex 不是最终结果——在设备上,dex 文件被重新编译成另一种形式 (.oat)。我会尝试的是确保一些非死代码实际上调用了注入对象的方法。 @MikhailNaganov 我也试过了。不幸的是没有区别。 【参考方案1】:

我之前也遇到过同样的问题,WebView只能通过android studio安装的和设备版本相同的SDK才能识别方法。在网上搜索后,我发现了这个:

Android WebView JavaScript callbacks fail in APK without Proguard

您是否已公开您的 JavascriptInterface 方法?我看到您的示例方法是公开的,但您没有显示真实代码,也许您在真实代码中忘记了这一点?

【讨论】:

以上是关于WebView JavaScript 接口似乎只能通过 Android Studio 的 Instant-Run 工作,但不能通过常规 gradle 编译的主要内容,如果未能解决你的问题,请参考以下文章

Android WebView

如何为javascript接口配置proguard?

带有 JavaScript 界面的 Webview

webview之总结2

使用 Delphi XE2 在 MacOS 中不显示 Webview

在 WebView 中注入 Javascript 桥接器