作者:fbysss
关键字:ClassLoader

一、Sun JDK Classloader体系

1.层次关系

Bootstrap: 加载rt.jar

<-Extension:加载 jre/ext/*.jar 

<-System:加载CLASSPATH中的jar

<-UserDefined加载用户自定义包类

2.特殊点

   Bootstrap ClassLoader为C++所写,Extension ClassLoader的parent为null,其中,getParent返回实例化本ClassLoader的ClassLoader,并不是superClass。

3.上级委托

很反感“双亲委托”这个翻译。给人第一印象要两个类去委托,而且,很容易感觉是继承关系。上级委托多好,立马减轻了理解难度。

上级委托的优点:

a.避免父子重复加载

b.避免jdk核心类被替代。比如String.class,在rt.jar包中。如果用户自己写了一个恶意代码,

由此想象一下我们经常遇到的,不同版本的jar包,出现冲突的原因(未经测试):

testv1.jar

testv2.jar

a.如果两个包是独立的,不被其他包中的class引用:

   如果两个包中的文件个数和名称完全一致,不会有冲突。因为同样的版本只加载一次。

   如果v2中有新的类,而加载的是testv1.jar,v2的新类就可能使用到v1的其他旧类,导致方法找不到或者逻辑错乱。

   如果加载的是testv2.jar,而v2是向下兼容,只增加类和方法,没有冲突。因为v1的类都不加载

   如果加载的是testv2.jar,只是删除了旧版本一些相对独立的类,这样旧版本中的这些类会被加载,并可以正常使用。

b.如果test包还有其他类引用,那就不好说了。类似上述情况,可能会出现冲突。

  

二、源码理解

protected synchronized Class loadClass(String name, boolean resolve)   
throws ClassNotFoundException   
   {   
// 首先检查该name指定的class是否有被加载   
Class c = findLoadedClass(name);   
if (c == null) {   
    try {   
    if (parent != null) {   
        //如果parent不为null,则调用parent的loadClass进行加载   
 = parent.loadClass(name, false);   
    } else {   
        //parent为null,则调用BootstrapClassLoader进行加载   
        c = findBootstrapClass0(name);   
    }   
    } catch (ClassNotFoundException e) {   
        //如果仍然无法加载成功,则调用自身的findClass进行加载               
        c = findClass(name);   
    }   
}   
if (resolve) {   
    resolveClass(c);   
}   
return c;   
   }

从这里也可以了解,ClassNotFoundException是因为什么情况产生了。就是当前类的类加载器没有找到对应jar或class文件。

Class.forname(String clazzName)实际上是使用了一个默认的类加载器,

public static Class<?> forName(String className) 
                throws ClassNotFoundException {
        return forName0(className, true, ClassLoader.getCallerClassLoader());
    }

从这里可以看出,使用的是调用Class.forname这个类所被加载的类加载器,对于一般的应用,就是SystemClassLoader,sun.misc.Launcher$AppClassLoader是一个实现

 

工具代码:用于查看Extension ClassLoader所加载的包

try {
   URL[] extURLs = ((URLClassLoader) ClassLoader
     .getSystemClassLoader().getParent()).getURLs();

   for (int i = 0; i < extURLs.length; i++) {

    System.out.println(extURLs[i]);

   }

  } catch (Exception e) {
  }

其他:

JVM中,一个类是否装载,由其fully qualified class name,也就是加包名的完整类名如com.sss.test.Test,及其装载器唯一决定。

也就是说,即使是同一个类,如果装载器不同,就可能被装载多次。

后记:

本想就写一篇笔记,但ClassLoader涉及的内容很多,还是一点一点来吧。