注意,我们清晰的看见这个三个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; public class URL { private String path; public URL(String path) { this.path = path; } public String toString() { return this.path + " new Path"; } } package java.net; import java.net.*; public class TheSameClsDemo { /** * @param args */ public static void main(String[] args) { URL url = new URL("http://www.baidu.com"); System.out.println(url.toString()); } } |