测试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方法