Proguard - 找不到常见的超级 class / java.lang.VerifyError

Proguard - Can't find common super class / java.lang.VerifyError

我们正在开发桌面应用程序。当运行 ProGuard(版本5.3.3)在代码上使用以下配置标志时:

-dontoptimize
-allowaccessmodification
-dontusemixedcaseclassnames
-dontwarn

Proguard 给出以下错误:

Unexpected error while performing partial evaluation:
  Class       = [com/code/backend/e/b/b]
  Method      = [b(Ljava/lang/String;)Ljava/nio/file/Path;]
  Exception   = [java.lang.IllegalArgumentException] (Can't find common super class of [com/google/common/collect/ImmutableList] (with 2 known super classes) and [com/google/common/collect/UnmodifiableIterator] (with 1 known super classes))
Unexpected error while preverifying:
  Class       = [com/code/backend/e/b/b]
  Method      = [b(Ljava/lang/String;)Ljava/nio/file/Path;]
  Exception   = [java.lang.IllegalArgumentException] (Can't find common super class of [com/google/common/collect/ImmutableList] (with 2 known super classes) and [com/google/common/collect/UnmodifiableIterator] (with 1 known super classes))


FAILURE: Build failed with an exception.

* What went wrong:
Execution failed for task 'proguard'.
> Can't find common super class of [com/google/common/collect/ImmutableList] (with 2 known super classes) and [com/google/common/collect/UnmodifiableIterator] (with 1 known super classes)

如果将标志 -dontpreverify 添加到配置中,构建将会成功。但是,当 运行 jar 文件时,我们会收到以下错误:

Error: A JNI error has occurred, please check your installation and try again
Exception in thread "main" java.lang.VerifyError: Expecting a stackmap frame at branch target 11
Exception Details:
  Location:
    com/code/code2/MainWindow.b(Lcom/code/code2/ClientAPIProtos$SoftwareStatus;)V @1: ifnull
  Reason:
    Expected stackmap frame at this location.
  Bytecode:
    0x0000000: 2bc6 000a 2ab4 00bc c700 04b1 2bb2 00a4
    0x0000010: a500 0a2b b800 f599 0018 2ab4 00bc b601
    0x0000020: 9299 0020 2ab4 00bc 03b6 0193 a700 152a
    0x0000030: b400 bcb6 0192 9a00 0b2a b400 bc04 b601
    0x0000040: 932b b800 f699 0012 1225 4db8 0138 4e2a
    0x0000050: 04b5 00c8 a700 0f12 1e4d b801 3a4e 2a03
    0x0000060: b500 c82a b400 bcb6 0191 3a04 1904 c600
    0x0000070: 0c2c 1904 b601 799a 000b 2ab4 00bc 2cb6
    0x0000080: 0195 2ab4 00bc b601 903a 0519 05c6 0009
    0x0000090: 2d19 05a5 000b 2ab4 00bc 2db6 0194 b1  

    at java.lang.Class.getDeclaredMethods0(Native Method)
    at java.lang.Class.privateGetDeclaredMethods(Class.java:2701)
    at java.lang.Class.privateGetMethodRecursive(Class.java:3048)
    at java.lang.Class.getMethod0(Class.java:3018)
    at java.lang.Class.getMethod(Class.java:1784)
    at sun.launcher.LauncherHelper.validateMainClass(LauncherHelper.java:544)
    at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:526)

这可以通过使用 JVM 标志 -noverify 来避免。阅读主题:

Obfuscation causes VerifyError: Expecting a stackmap frame

我们希望尽可能避免这些 'no verification' 标记。

注意:即使添加

也会出现构建错误
-dontobfuscate
-dontshrink

初始化、验证和验证是构建项目的一部分。我相信标志 -allowaccessmodification 是允许在构建项目时进行修改(即你的 classes 是从原始结构修改的)。当原始 class 结构被修改时,它们的栈帧就会被扰乱。这导致验证失败。如果您想忽略验证错误,那么如前所述,您需要使用暂停验证的标志。

您正在指定选项 -dontwarn 以禁止显示有关未解析引用的警告。这可能会导致在处理代码时出现问题。 ProGuard 需要这些引用,例如在预验证步骤中。该错误确实表明输入不包含所有必要的依赖项:缺少 ImmutableListUnmodifiableIterator 的一些公共超类或接口。对于桌面应用程序,您应该检查您指定的 Java 运行时 jar 是否包含所有必要的 类:

-libraryjars <java.home>/lib/rt.jar

参见 ProGuard 手册 > 疑难解答 > Warning: can't find superclass or interface

Troubleshooting

ProGuard 可能会打印出一些注释和非致命警告:

注意:找不到动态引用的class ProGuard 无法通过内省找到您的代码正在访问的 class 或接口。您应该检查是否要添加包含此 class.

的 jar

注意:...调用“(...)Class.forName(variable).newInstance()” ProGuard 列出了动态创建的 class 实例的所有 class 转换,例如“(MyClass)Class.forName(variable).newInstance()”。根据您的应用程序,您可能需要使用 -keep class MyClass 等选项保留上述 classes,或者使用 -keep class * implements MyClass 等选项保留它们的实现。您可以通过指定 -dontnote 选项关闭这些注释。

注意:...动态访问 field/method“...” ProGuard 列出了许多构造,例如“.getField("myField")”。根据您的应用程序,您可能需要找出所提到的 class 成员的定义位置,并使用 -keep class MyClass { MyFieldType myField; } 等选项保留它们。否则,ProGuard 可能会删除或混淆 class 成员,因为它不知道它们到底是哪些。它确实列出了可能的候选人,供您参考。您可以通过指定 -dontnote 选项关闭这些注释。

manual/troubleshooting 作为最终解决方案,您可以关闭优化 (-dontoptimize) 和预验证 (-dontpreverify)。