Proguard - 找不到常见的超类/java.lang.VerifyError
Posted
技术标签:
【中文标题】Proguard - 找不到常见的超类/java.lang.VerifyError【英文标题】:Proguard - Can't find common super class / java.lang.VerifyError 【发布时间】:2018-01-06 18:42:09 【问题描述】:我们正在开发一个桌面应用程序。在使用以下配置标志的代码上运行 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
如果可能,我们希望避免使用这些“无验证”标志。
注意:即使添加也会出现构建错误
-dontobfuscate
-dontshrink
【问题讨论】:
【参考方案1】:您正在指定选项 -dontwarn
以禁止显示有关未解析引用的警告。这可能会导致处理代码时出现问题。 ProGuard 需要这些参考,例如在预验证步骤中。该错误确实表明输入不包含所有必要的依赖项:缺少ImmutableList
和UnmodifiableIterator
的一些公共超类或接口。对于桌面应用程序,您应该检查是否指定了包含所有必要类的 Java 运行时 jar:
-libraryjars <java.home>/lib/rt.jar
请参阅 ProGuard 手册 > 疑难解答 > Warning: can't find superclass or interface
【讨论】:
此链接失效【参考方案2】:初始化、验证和确认是构建项目的一部分。我相信标志-allowaccessmodification
是允许在构建项目时进行修改(即您的类是从原始结构修改的)。当原始类结构被修改时,它们的堆栈帧就会受到干扰。这会导致验证失败。如果您想忽略验证错误,那么如上所述,您需要使用暂停验证的标志。
【讨论】:
【参考方案3】:Troubleshooting
ProGuard 可能会打印出一些注释和非致命警告:
注意:找不到动态引用的类 ProGuard 找不到您的代码通过自省方式访问的类或接口。您应该检查是否要添加包含此类的 jar。
注意:...调用 '(...)Class.forName(variable).newInstance()'
ProGuard 列出了动态创建的类实例的所有类转换,例如“(MyClass)Class.forName(variable).newInstance()”。根据您的应用程序,您可能需要使用-keep class MyClass
之类的选项保留提到的类,或者使用-keep class * implements MyClass
之类的选项保留它们的实现。您可以通过指定-dontnote
选项来关闭这些注释。
注意:...动态访问字段/方法“...”
ProGuard 列出了许多结构,例如“.getField("myField")”。根据您的应用程序,您可能需要确定上述类成员的定义位置,并使用-keep class MyClass MyFieldType myField;
之类的选项保留它们。否则,ProGuard 可能会删除或混淆类成员,因为它无法知道它们到底是哪些成员。它确实列出了可能的候选人,供您参考。您可以通过指定-dontnote
选项来关闭这些注释。
【讨论】:
【参考方案4】:manual/troubleshooting 作为最终解决方案,您可以关闭优化 (-dontoptimize) 和预验证 (-dontpreverify)。
【讨论】:
以上是关于Proguard - 找不到常见的超类/java.lang.VerifyError的主要内容,如果未能解决你的问题,请参考以下文章
从 Proguard 获取“未解决的引用”和“找不到超类”警告
ProGuard 错误找不到超类或接口 org.apache.http.entity
找不到“AFURLSessionManager”的接口声明,“AFHTTPSessionManager”的超类?
`attemptRecovery(fromError:optionIndex:)` 在 NSDocument 的 Swift 子类的超类中找不到