问题描述
今天导入了同事的包,启动报错
java.lang.NoSuchMethodError: com.szkingdom.jtup.util.HttpUtil.sendPost(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
很奇怪的是,使用Ctrl方法是可以点击跳转,工程中也是使用了该方法,在排除该方法不存在的情况,全局搜索才发现原来是其他的jar的包命名和类命名是一样的导致的,索性就改了类名,该bug得到了解决;
1.原因
java 的类加载机制是把所有不同名称的本类和引用类的包全部加载到内存,这样就有一个问题,如果有两个类在JVM看来是完全一样的(包名和类名完全一样,即全限定名一样)怎么办,机器无法区分
举个栗子,
引用类,即引用别人包中的类(其包名:com.qq.dto):
//引用类
public class HttpUtil(){
public static sendPost(){
System.out.println("引用类");
}
}
public class (){
public static sendPost(){
System.out.println("本类");
}
public static getDataOut(){
System.out.println("本类");
}
}
上述两个类在JVM看来是完全一样的(全限定名一样),JVM 只会加载一个类。如果先加载引用类,在调用 HttpUtil.sendPost 方法时,机器肯定先去使用引用类的 HttpUtil.sendPost 方法,可实际需要调用本类的方法,这样肯定就会出错,如果调用想调用本类的HttpUtil.getDataOut方法,因为内存加载的是引用类,就会发生找不到方法的错误。
2.解决
最简单的解决就是全限定名不一样(改包名或类名),这样就可以完全区分开来。
可以用 mvn dependency:analyze,mvn dependency:tree 命令辅助分析加载包的顺序。