设置
我有一个托管 WebView 组件的简单片段。在这个片段的初始化过程中,我向我的应用程序注册了一些 JavaScript 接口方法:
mJavaScriptHooks = new JavaScriptHooks();
...
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.addJavascriptInterface(mJavaScriptHooks, "Android");
我的 JavaScript 钩子并没有什么特别之处,所有这些钩子都以下列方式声明:
public class JavaScriptHooks {
@JavascriptInterface
public void someMethod() {
...
}
...
}
为了确保这些方法不会被混淆或优化而被遗忘,我在我的ProGuard 配置中添加了以下内容:
-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 时代)。