Java ClassLoader 原理详细分析
作者:网络转载 发布时间:[ 2015/3/24 11:11:46 ] 推荐标签:Java ClassLoader JDK
测试2:将ClassLoaderTest.class打包成ClassLoaderTest.jar,放到Extension ClassLoader的加载目录下(JAVA_HOME/jre/lib/ext),然后重新运行这个程序,得到的结果会是什么样呢?

打印结果:

打印结果分析:
为什么第一行的结果是ExtClassLoader呢?
因为ClassLoader的委托模型机制,当我们要用ClassLoaderTest.class这个类的时候,AppClassLoader在试图加载之前,先委托给Bootstrcp ClassLoader,Bootstracp ClassLoader发现自己没找到,它告诉ExtClassLoader,兄弟,我这里没有这个类,你去加载看看,然后Extension ClassLoader拿着这个类去它指定的类路径(JAVA_HOME/jre/lib/ext)试图加载,唉,它发现在ClassLoaderTest.jar这样一个文件中包含ClassLoaderTest.class这样的一个文件,然后它把找到的这个类加载到内存当中,并生成这个类的Class实例对象,后把这个实例返回。所以ClassLoaderTest.class的类加载器是ExtClassLoader。
第二行的结果为null,是因为ExtClassLoader的父类加载器是Bootstrap ClassLoader。
测试3:用Bootstrcp ClassLoader来加载ClassLoaderTest.class,有两种方式:
1、在jvm中添加-Xbootclasspath参数,指定Bootstrcp ClassLoader加载类的路径,并追加我们自已的jar(ClassTestLoader.jar)
2、将class文件放到JAVA_HOME/jre/classes/目录下(上面有提到)
方式1:(我用的是Eclipse开发工具,用命令行是在java命令后面添加-Xbootclasspath参数)
打开Run配置对话框:

配置好如图中所述的参数后,重新运行程序,产的结果如下所示:(类加载的过程,只摘下了一部份)
打印结果:
[Loaded java.io.FileReader from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.nio.cs.StreamDecoder from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.ArrayList from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.lang.reflect.Array from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.Locale from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.ConcurrentMap from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.ConcurrentHashMap from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.Lock from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.ReentrantLock from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.ConcurrentHashMap$Segment from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.AbstractOwnableSynchronizer from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.AbstractQueuedSynchronizer from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.ReentrantLock$Sync from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.ReentrantLock$NonfairSync from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.locks.AbstractQueuedSynchronizer$Node from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.concurrent.ConcurrentHashMap$HashEntry from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.lang.CharacterDataLatin1 from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.io.ObjectStreamClass from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.net.www.ParseUtil from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.BitSet from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.net.Parts from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.net.URLStreamHandler from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.net.www.protocol.file.Handler from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.util.HashSet from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.net.www.protocol.jar.Handler from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.misc.Launcher$AppClassLoader from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded sun.misc.Launcher$AppClassLoader$1 from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.lang.SystemClassLoaderAction from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Path C:Program FilesJavajdk1.6.0_22jreclasses]
[Loaded classloader.ClassLoaderTest from C:Program FilesJavajdk1.6.0_22jreclasses]
null//这是打印的结果
C:Program FilesJavajdk1.6.0_22jrelib
esources.jar;C:Program FilesJavajdk1.6.0_22jrelib
t.jar;
C:Program FilesJavajdk1.6.0_22jrelibsunrsasign.jar;C:Program FilesJavajdk1.6.0_22jrelibjsse.jar;
C:Program FilesJavajdk1.6.0_22jrelibjce.jar;C:Program FilesJavajdk1.6.0_22jrelibcharsets.jar;
C:Program FilesJavajdk1.6.0_22jreclasses;c:ClassLoaderTest.jar
//这一段是System.out.println(System.getProperty("sun.boot.class.path"));打印出来的。这个路径是Bootstrcp ClassLoader默认搜索类的路径
[Loaded java.lang.Shutdown from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
[Loaded java.lang.Shutdown$Lock from C:Program FilesJavajdk1.6.0_22jrelib
t.jar]
方式2:将ClassLoaderTest.jar解压后,放到JAVA_HOME/jre/classes目录下,如下图所示:
提示:jre目录下默认没有classes目录,需要自己手动创建一个

打印结果:

从结果中可以看出,两种方式都实现了将ClassLoaderTest.class由Bootstrcp ClassLoader加载成功了。
四、定义自已的ClassLoader
既然JVM已经提供了默认的类加载器,为什么还要定义自已的类加载器呢?
因为Java中提供的默认ClassLoader,只加载指定目录下的jar和class,如果我们想加载其它位置的类或jar时,比如:我要加载网络上的一个class文件,通过动态加载到内存之后,要调用这个类中的方法实现我的业务逻辑。在这样的情况下,默认的ClassLoader不能满足我们的需求了,所以需要定义自己的ClassLoader。
定义自已的类加载器分为两步:
1、继承java.lang.ClassLoader
2、重写父类的findClass方法

sales@spasvo.com