3 回答
TA贡献1802条经验 获得超4个赞
Java 虚拟机的第一个类加载器是 Bootstrap,这个加载器是嵌套在 Java 虚拟机内核里面,它是用 C++ 写的二进制代码(不是字节码)。
使用委托机制,会递归的向父类查找,也就是首选用 Bootstrap 尝试加载,如果找不到再向下。防止内存中出现两份类的字节码。
你误解了类和类加载器。
当类记载的时候,首先用当前线程的类加载器去加载线程中的第一个类,比如这个类是 ClassA,类加载器是 ClassLoaderA。
如果 ClassA 引用了 ClassB,则系统会使用 ClassLoaderA 去加载 ClassB。
现在有了 2 个类(简化版,其实不止是 2 个类)。
我们考虑一种情况,ClassX 和 ClassY 已经在内存中加载,他们都引用了 ClassZ,那么 ClassZ 由谁加载呢?
很显然按照上面描述的加载步骤会出现 2 份 ClassZ:ClassX 加载一次,ClassY 又加载一次。因为 ClassY 不知道 ClassX 已经加载过了。
如何解决这个问题呢,就是向父类递归查找。
具体步骤就是,先从 BootstrapClassLoader 查找,如果 BootstrapClassLoader 加载了这个类,就返回,如果 BootstrapClassLoader 没有加载过这个类,则继续查找,直到找到这个类。如果一直找到了本线程的类加载器都没有找到,说明这个类还没有加载,则使用当前线程的加载器加载。可以使用 getContextClassLoader()
获得当前线程的类加载器。
TA贡献1848条经验 获得超2个赞
Java中有两类类加载器:系统类加载器和用户自定义类加载器。
系统类加载器都会有加载路径的限定,比如Bootstrap Class Loader
在JDK1.6下,通过System.getProperty("sun.boot.class.path")
可以得到类加载路径
JAVA_HOME\jre6\lib\resources.jar;
JAVA_HOME\Java\jre6\lib\rt.jar;
JAVA_HOME\jre6\lib\sunrsasign.jar;
JAVA_HOME\jre6\lib\jsse.jar;JAVA_HOME\jre6\lib\jce.jar;
JAVA_HOME\jre6\lib\charsets.jar;
JAVA_HOME\jre6\lib\modules\jdk.boot.jar;
JAVA_HOME\jre6\classes
这些路径下的class是由Bootstrap
负责,其它路径下的class的递归到Bootstrap
下也是找不到class文件,就会由下一级类加载器去相应的路径去加载。
添加回答
举报