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 编译的主要内容,如果未能解决你的问题,请参考以下文章