例1,测试你所使用的JVM的ClassLoader
/*LoaderSample1.java*/ public class LoaderSample1 { public static void main(String[] args) { Class c; ClassLoader cl; cl = ClassLoader.getSystemClassLoader(); System.out.println(cl); while (cl != null ) { cl = cl.getParent(); System.out.println(cl); } try { c = Class.forName( " java.lang.Object " ); cl = c.getClassLoader(); System.out.println( " java.lang.Object's loader is " + cl); c = Class.forName( " LoaderSample1 " ); cl = c.getClassLoader(); System.out.println( " LoaderSample1's loader is " + cl); } catch (Exception e) { e.printStackTrace(); } } } |
在我的机器上(Sun Java 1.4.2)的运行结果
sun.misc.Launcher$AppClassLoader@1a0c10f
sun.misc.Launcher$ExtClassLoader@e2eec8
null
java.lang.Object's loader is null
LoaderSample1's loader is sun.misc.Launcher$AppClassLoader@1a0c10f
- 第一行表示,系统类装载器实例化自类sun.misc.Launcher$AppClassLoader
- 第二行表示,系统类装载器的parent实例化自类sun.misc.Launcher$ExtClassLoader
- 第三行表示,系统类装载器parent的parent为bootstrap
- 第四行表示,核心类java.lang.Object是由bootstrap装载的
- 第五行表示,用户类LoaderSample1是由系统类装载器装载的
注意,我们清晰的看见这个三个ClassLoader类之间的父子关系(不是继承关系),父子关系在ClassLoader的实现中有一个ClassLoader类型的属性,我们可以在自己实现自定义的ClassLoader的时候初始化定义,而这三个系统定义的ClassLoader的父子关系分别是
AppClassLoader—————》(Parent)ExtClassLoader————————》(parent)BootClassLoader(null c++实现)
系统为什么要分别指定这么多的ClassLoader类呢?
答案在于因为java是动态加载类的,这样的话,可以节省内存,用到什么加载什么,就是这个道理,然而系统在运行的时候并不知道我们这个应用与需要加载些什么类,那么,就采用这种逐级加载的方式
(1)首先加载核心API,让系统最基本的运行起来
(2)加载扩展类
(3)加载用户自定义的类
package org.corey.clsloader; import java.net.MalformedURLException; import java.net.URL; import java.net.URLClassLoader; import sun.net.spi.nameservice.dns.DNSNameService; public class ClsLoaderDemo { /** * @param args */ public static void main(String[] args) { System.out.println(System.getProperty("sun.boot.class.path")); System.out.println(System.getProperty("java.ext.dirs")); System.out.println(System.getProperty("java.class.path")); } } |
程序结果为:
E:\MyEclipse 6.0\jre\lib\rt.jar;E:\MyEclipse 6.0\jre\lib\i18n.jar;E:\MyEclipse 6.0\jre\lib\sunrsasign.jar;E:\MyEclipse 6.0\jre\lib\jsse.jar;E:\MyEclipse 6.0\jre\lib\jce.jar;E:\MyEclipse 6.0\jre\lib\charsets.jar;E:\MyEclipse 6.0\jre\classes
E:\MyEclipse 6.0\jre\lib\ext
E:\workspace\ClassLoaderDemo\bin
在上面的结果中,你可以清晰看见三个ClassLoader分别加载类的路径;也知道为什么我们在编写程序的时候,要把用到的jar包放在工程的classpath下面啦,也知道我们为什么可以不加载java.lang.*包啦!其中java.lang.*就在rt.jar包中;
(4)ClassLoader的加载机制:
现在我们设计这种一下Demo:
package java.net; |