多个不同的GAV-classifier依赖冲突,引起NoSuchMethodError
Maven依赖的三坐标体系GAV(G-groupId,A-artifactId,V-version)
classifier
通常用于区分从同一POM构建的具有不同内容的构件物(artifact)。它是可选的,它可以是任意的字符串,附加在版本号之后。
【案例1】某应用依赖两个GAV-classifier不同的snakeyaml.jar,引起NoSuchMethodError
1.完整的异常调用栈【从案发现场找线索】
警察抓犯人,收集线索是破案的关键所在。
2.关键的异常日志
从异常信息中收集有用的线索。
java.lang.NoSuchMethodError: org.yaml.snakeyaml.constructor.SafeConstructor.<init>(Lorg/yaml/snakeyaml/LoaderOptions;)Vat org.apache.dubbo.rpc.cluster.router.condition.config.model.ConditionRuleParser.parse(ConditionRuleParser.java:44)
java.lang.NoSuchMethodError
that class no longer has a definition of that method.
该类不再具有该方法的定义。
3.案发现场的源代码
org.apache.dubbo.rpc.cluster.router.condition.config.model.ConditionRuleParser.parse(ConditionRuleParser.java:44)
- jad - 反编译指定已加载类的源码
[arthas@41915]$ jad org.apache.dubbo.rpc.cluster.router.condition.config.model.ConditionRuleParser parse
ConditionRuleParser.java:44
,与异常信息对上了。
org.yaml.snakeyaml.constructor.SafeConstructor
点击链接到snakeyaml-1.23-android.jar
4.应用里依赖两个snakeyaml.jar是怎么回事?
Maven: org.yaml:snakeyaml:1.26,依赖管理声明自spring-boot-dependencies
Maven: org.yaml:snakeyaml:android:1.23,从其他依赖透明传递下来
5.使用Arthas命令确认已加载的类信息和方法信息
class/classloader 相关
破案杀手锏
- sc - 查看 JVM 已加载的类信息
- sm - 查看已加载类的方法信息
[arthas@41915]$ sc -d org.yaml.snakeyaml.constructor.SafeConstructorclass-info org.yaml.snakeyaml.constructor.SafeConstructorcode-source /Users/lihuagang/.m2/repository/org/yaml/snakeyaml/1.23/snakeyaml-1.23-android.jarname org.yaml.snakeyaml.constructor.SafeConstructorisInterface falseisAnnotation falseisEnum falseisAnonymousClass falseisArray falseisLocalClass falseisMemberClass falseisPrimitive falseisSynthetic falsesimple-name SafeConstructormodifier publicannotationinterfacessuper-class +-org.yaml.snakeyaml.constructor.BaseConstructor+-java.lang.Objectclass-loader +-sun.misc.Launcher$AppClassLoader@18b4aac2+-sun.misc.Launcher$ExtClassLoader@4d95d2a2classLoaderHash 18b4aac2Affect(row-cnt:2) cost in 34 ms.[arthas@41915]$ sm org.yaml.snakeyaml.constructor.SafeConstructor <init>
org.yaml.snakeyaml.constructor.SafeConstructor <init>()VAffect(row-cnt:2) cost in 55 ms.[arthas@41915]$ sm -d org.yaml.snakeyaml.constructor.SafeConstructor <init>declaring-class org.yaml.snakeyaml.constructor.SafeConstructorconstructor-name <init>modifier publicannotationparametersexceptionsclassLoaderHash 18b4aac2
- 通过
sc -d org.yaml.snakeyaml.constructor.SafeConstructor
,可以看到其加载自snakeyaml-1.23-android.jar
。 - 通过
sm org.yaml.snakeyaml.constructor.SafeConstructor <init>
,可以看到没有org.yaml.snakeyaml.constructor.SafeConstructor#SafeConstructor(org.yaml.snakeyaml.LoaderOptions)
构造函数的方法签名。 - 综上所述,与异常信息完全符合。
6.解决方法
snakeyaml-1.23-android.jar
依赖排除掉,搞定。
祝大家玩得开心!ˇˍˇ
广益,杭州