Java类加载器作用:
将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区中的运行时数据结构,在堆中生成一个代表这个类的java.lang.Class对象,作为方法区类数据的访问入口。
类加载器层次结构(树状结构)
- 引导类加载器(bootstrap class loader)
- 用来加载Java的核心库(JAVA_HOME/jre/lib/rt.jar或sun.boot.class.path路径下的内容),是用原生代码实现的,并不继承自java.lang.ClassLoader
- 加载扩展类和应用程序类加载器。并指定它们的父类加载器
- 扩展类加载器(extensions class loader)
- 用来加载Java扩展库(Java_Home/jre/ext/*.jar或java.ext.dirs路径下的内容)。Java虚拟机的实现会提供一个扩展库目录,该类加载器在此目录里面查找并加载java类
- 由sun.misc.Launcher$ExClassLoader实现
- 应用程序类加载器(application class loader)
- 根据Java应用的类路径(classpath, java.class.path)一般来说,Java应用的类都是由它加载完成的
- 由sun.misc.Launcher$AppClassLoader实现
- 自定义类加载器
- 开发人员可以通过集成java.lang.ClassLoader类的方式实现自己的类加载器,以满足一些特殊需求
自定义类加载器 父亲》应用程序类加载器 父亲》扩展类加载器 父亲》引导类加载器
但他们不是继承关系,内部组织结构之间采用组合来实现。逻辑上仍然是父子关系。而除了引导类加载器外其他三种加载器都继承了java.lang.ClassLoader, 引导类加载器是由C++实现的
java.lang.ClassLoader类的基本职责就是根据一个指定的类的名称找到或者生成对应的字节代码,然后从这些字节代码中定义出一个Java类,即java.lang.Class类的一个实例。此外,还负责加载Java应用所需的资源,如图像文件和配置方法等。
可以通过
System.out.println(System.getProperty("java.class.path"));
查看classpath加载类的位置。
public class Demo02 {
public static void main(String[] args) {
// TODO Auto-generated method stub
//sun.misc.Launcher$AppClassLoader@73d16e93
System.out.println(ClassLoader.getSystemClassLoader());
//sun.misc.Launcher$ExtClassLoader@15db9742
System.out.println(ClassLoader.getSystemClassLoader().getParent());
//null 应该是引导类加载器,不过引导类加载器由原生代码实现,因此显示为null
System.out.println(ClassLoader.getSystemClassLoader().getParent().getParent());
}
}
代理模式:交给其他类来加载指定的类
双亲委托机制: 代理模式的一种。Java的应用程序类加载器和扩展类加载器采用这种模式
- 某个特定的类加载器在接受到加载类的请求时,首先将加载任务委托给父类加载器,依次回溯,直到最高的爷爷辈,如果父类加载器可以完成类加载的任务,就成返回;只有父类加载器无法完成此加载任务时,才会自己去加载
- 双亲委托机制是为了保证Java核心库的类型安全。这种机制就可以保证不会出现用户能自定义java.lang.object累的情况。
- 类加载机制除了用于加载类,也是安全的最基本的屏障
并不是所有的类加载器都采用双亲委托机制
双亲委托机制测试
1、新建java.lang包,在包中新建String类
package java.lang;
public class String {
public String toString() {
return "helllo";
}
}
2、Demo02测试该String执行效果
package com.feng.ClassTest;
public class Demo02 {
public static void main(String[] args) {
// TODO Auto-generated method stub
//sun.misc.Launcher$AppClassLoader@73d16e93
System.out.println(ClassLoader.getSystemClassLoader());
//sun.misc.Launcher$ExtClassLoader@15db9742
System.out.println(ClassLoader.getSystemClassLoader().getParent());
//null 应该是引导类加载器,不过引导类加载器由原生代码实现,因此显示为null
System.out.println(ClassLoader.getSystemClassLoader().getParent().getParent());
String a = "123";
System.out.println(a); //123
System.out.println(a.getClass().getClassLoader()); //null
}
}
null是因为该类被引导类加载器加载,加载的是Java核心库中String类,而不是自定义的String类。
注:loadClass