最近工作中遇到了一个项目工程问题,在启动jvm的classpath有两个不同版本的jwt的jar包,在调用处报java.lang.NoSuchMethodError:
其Classpath有两个不同版本的jar包,里面都有这个类,高版本的jar里面没有这个Method,低版本有这个Method。最终没有加载这个高版本的Method
猜测此问题就是 “全限定类名” 完全一样,Cloassloader只加载了高版本的jar包。
此文就是为了验证jvm是如何加载classpath中同类同全限定类名的过程。
准备三个不同的jar,里面都有同样一个类 Car
,如下:
这三个工程拥有同样一个class: com.example.demo.Car
public class Car {
private static final String version = "A4L";
public String getVersion() {
return version;
}
public String getName() {
return "audi";
}
public Integer limitSpeed() {
return 100;
}
public String seatPerson(Integer a) {
return "可以坐 :" + a;
}
}
public class Car {
private static final String version = "A6L";
public String getVersion() {
return version;
}
public String getName() {
return "audi";
}
public Integer limitSpeed() {
return 140;
}
public String seatPerson() {
return "可以坐 :5";
}
}
public class Car {
private static final String version = "E300L";
public String getVersion() {
return version;
}
public String getName() {
return "mercedes";
}
public Integer limitSpeed() {
return 135;
}
public String hasPerson() {
return "能舒服的坐 :4";
}
}
example.java
public class Example {
public static void main(String[] args) {
Car car = new Car();
System.out.println("当前车辆版本:" + car.getVersion());
System.out.println("当前 jar 包路径 : ");
System.out.println(car.getClass().getProtectionDomain().getCodeSource().getLocation().getPath());
Method[] declaredMethods = car.getClass().getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
System.out.println("------------------");
System.out.println("method name: " + declaredMethod.getName());
List<String> collect = Arrays.stream(declaredMethod.getParameterTypes()).map(Class::getName).collect(Collectors.toList());
if(!collect.isEmpty()) {
System.out.println("parameter type : " + collect);
}
System.out.println("------------------");
}
}
}
test 目录
demo-audi-1.0.jar demo-audi-2.0.jar demo-example-1.0.jar demo-mercedes-1.0.jar
java -classpath "/test/*" com.example.demo.Example
当前车辆版本:A4L
当前 jar 包路径 :
/test/demo-audi-1.0.jar
------------------
method name: getName
------------------
------------------
method name: getVersion
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: seatPerson
parameter type : [java.lang.Integer]
------------------
运行时,JVM加载类Car,根据【操作系统】的选择,本次加载了demo-audi-1.0.jar的com.example.demo.Car
类的class文件
当把 demo-audi-1.0.jar 重命名为 zemo-audi-1.0.jar ,“d” 改成 “z”
➜ test mv demo-audi-1.0.jar zemo-audi-1.0.jar
➜ test java -classpath "/test/*" com.example.demo.Example
当前车辆版本:A6L
当前 jar 包路径 :
/test/demo-audi-2.0.jar
------------------
method name: getName
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: getVersion
------------------
------------------
method name: seatPerson
------------------
运行时,JVM加载类Car,根据【操作系统】的选择,本次加载了demo-audi-2.0.jar的com.example.demo.Car
类的class文件
当把 demo-audi-2.0.jar 重命名为 demo-mercedes-2.0.jar ,“audi” 改成 “mercedes”
[/test]# mv demo-audi-2.0.jar zemo-mercedes-2.0.jar
[/test]# ls demo-example-1.0.jar demo-mercedes-1.0.jar demo-mercedes-2.0.jar zemo-audi-1.0.jar
[/test]# java -classpath "/test/*" com.example.demo.Example
当前车辆版本:A6L
当前 jar 包路径 :
/test/demo-mercedes-2.0.jar
------------------
method name: getName
------------------
------------------
method name: getVersion
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: seatPerson
------------------
运行时,JVM加载类Car,根据【操作系统】的选择,本次加载了demo-mercedes-2.0.jar的com.example.demo.Car
类的class文件
如果把audi的两个jar移动出去,classpath里面只剩下 demo-mercedes-1.0.jar 时
当前车辆版本:E300L
当前 jar 包路径 :
/test/demo-mercedes-1.0.jar
------------------
method name: getName
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: getVersion
------------------
------------------
method name: hasPerson
------------------
运行时,JVM加载类Car,根据【操作系统】的选择,本次加载了demo-mercedes-1.0.jar的com.example.demo.Car
类的class文件
# 当 demo-audi-1.0.jar在第一个时
java -classpath /test/demo-audi-1.0.jar:/test/demo-audi-2.0.jar:/test/demo-mercedes-1.0.jar:/test/demo-example-1.0.jar com.example.demo.Example
当前车辆版本:A4L
当前 jar 包路径 :
/test/demo-audi-1.0.jar
------------------
method name: getName
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: getVersion
------------------
------------------
method name: seatPerson
parameter type : [java.lang.Integer]
# 当 demo-audi-2.0.jar在第一个时
java -classpath /test/demo-audi-2.0.jar:/test/demo-audi-1.0.jar:/test/demo-mercedes-1.0.jar:/test/demo-example-1.0.jar com.example.demo.Example
当前车辆版本:A6L
当前 jar 包路径 :
/test/demo-audi-2.0.jar
------------------
method name: getName
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: getVersion
------------------
------------------
method name: seatPerson
------------------
# 当 demo-mercedes-1.0.jar在第一个时
java -classpath /test/demo-mercedes-1.0.jar:/test/demo-audi-2.0.jar:/test/demo-audi-1.0.jar:/test/demo-example-1.0.jar com.example.demo.Example
当前车辆版本:E300L
当前 jar 包路径 :
/test/demo-mercedes-1.0.jar
------------------
method name: getName
------------------
------------------
method name: limitSpeed
------------------
------------------
method name: getVersion
------------------
------------------
method name: hasPerson
------------------
根据JVM的双亲委派模型,默认情况下相同全限定类名的类只会加载一次,因此JVM加载Car类时只会从demo-audi-1.0.jar或demo-audi-2.0.jar以及demo-mercedes-1.0.jar选一个;
同名的两个Car类来自不同的三个Jar包,他们是平级的,根据JVM的类加载机制——双亲委派模型,相同全限定类名的类默认只会加载一次(除非手动破坏双亲委派模型);
Jar包中的类是使用AppClassLoader加载的,而类加载器中有一个命名空间的概念,同一个类加载器下,相同包名和类名的class只会被加载一次,如果已经加载过了,直接使用加载过的;
如果依赖中有多个全限定类名相同的类,那JVM会加载哪一个类呢?
比较靠谱的说法是,操作系统本身,控制了Jar包的默认加载顺序;也就是说,对于我们来说是不明确不确定的!
而Jar包的加载顺序,是跟classpath这个参数有关,当使用idea启动springboot的服务时,可以看到classpath参数的;包路径越靠前,越先被加载;
换句话说,如果靠前的Jar包里的类被加载了,后面Jar包里有同名同路径的类,就会被忽略掉,不会被加载;