首先自己定义一个包叫做java.lang,然后在里面定义一个类叫做Object.
然后在里面定义一个方法叫做toString(),返回null.另一个叫sayHello()
package java.lang;
public class Object {
public String toString() {
return null;
}
public void sayHello() {
System.out.println("hello");
}
}
然后在另一个包里面使用Object对象:
package top.test;
import java.lang.Object;
public class ObjectTest {
public static void main(String[] args) {
Object o = new Object();
System.out.println(o.toString());
o.sayHello();
}
}
猜一猜这个Object会是哪个Object呢?
有人猜是自己写的哪个Object吗?因为o.sayHello()都调用成功了,Eclipse没有报错,那么肯定是自己写的哪个Object啊.但是当我们尝试去运行时,出现问题了
会报java.lang.NoSuchMethodError异常,说没有找到这个方法.
为什么?
这是因为java的类加载机制是基于”双亲委派模型”,看下面这个图:
java的类加载器分为4类,如图所示,当加载一个类的时候,如果我们没有定义自己的类加载器,默认是使用Application ClassLoader.除了Bootstrap ClassLoader以外,都有自己的父类加载器.
双亲委派模型的机制是这样的(这个机制不是强制的,自己的类加载器可以不遵守):
如果接收到了一个类加载请求,总是尝试着把这个请求传送给父类加载器,只有在父类加载器不能处理这个请求的时候,自己才尝试加载
实现这个逻辑需要在代码中自行实现,不是强制的.
解释我们程序的结果:
所以当我们加载自己写的java.lang.Object时,会默认调用Appliation ClassLoader,这是系统提供的类加载器,肯定支持”双亲委派模型”,所以我们的请求会一步步提交到Bootstrap ClassLoader那里,这个类默认加载的类位于$JAVA_HOME/jre/lib下面的rt.jar包,可以找到我们需要的java.lang.Object类,所以加载的自然就不是我们自己写的Object类了.
经测试,在Eclipse写出这样的代码不会报错,但是在Idea中会报错,看来还是Idea更强大一些.