类加载器是用来将指定类的字节码文件(.class)加载到JVM中,然后由JVM来进行转换成对应的机器码。每个java程序至少拥有三个类加载器,分别是:
1、引导类加载器(Bootstrap)
2、扩展类加载器(Extension)
3、系统类加载器或应用类加载器(System)
一、引导类加载器(Bootstrap)
引导类加载器是java虚拟机不可分割的一部分,而且通常是用C语言实现的。该类加载器通常从java_home/jre/lib/rt.jar中加载系统类。如:String、List等系统API。ClassLoader loader = String.class.getClassLoader();
System.out.println(loader); // null
注意:引导类加载器没有ClasssLoader对象,如上面的String就的ClassLoader对象等于null。
二、扩展类加载器(Extension)
扩展类加载器用于从java_home/jre/lib/ext目录加载“标准的扩展”。可以将JAR文件放入该目录,这样即使在classpath下面没有指定,扩展类加载也可以找到其中的各个类。有人推荐使用该机制来避免“可恶的CLASSPATH”,不过扩展类加载器并不使用classpath路径来加载类,即:在扩展目录下面的类使用了classpath下面的类会出现类没有找到错误信息。ClassLoader extLoader = Thread.currentThread().getContextClassLoader().getParent();
// extLoader 为 sun.misc.Launcher$ExtClassLoader
// 这里获取了当前线程的上下文类加载器,该类加载器的父类加载器就是扩展类加载器
三、系统类加载器(System)
系统类加载器用于加载应用类,它在由classpath环境变量或者java命令的-classpath或-cp选项设置的类路径中的目录或者jar/zip文件里面查找。ClassLoader appLoader = Thread.currentThread().getContextClassLoader();
// appLoader 为 sun.misc.Launcher$AppClassLoader
// 这里获取了当前线程的上下文类加载器,默认为启动该线程的线程上下文类加载器
实例代码:
创建两个DAO对象,分别是StudentDao(学生)和TeacherDao(教师),TeacherDao中包含StudentDao对象,我将TeacherDao打包成test.jar包,然后放置到java_home/jre/lib/ext目录下面,然后使用Class.forName来加载TeacherDao类且创建该类实例。如下图:
TeacherDao类代码:package com.bug315.dao;
public class TeacherDao {
private int id;
private String name;
private StudentDao student;
public TeacherDao() {
}
public TeacherDao(int id, String name, StudentDao student) {
this.id = id;
this.name = name;
this.student = student;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public StudentDao getStudent() {
return student;
}
public void setStudent(StudentDao student) {
this.student = student;
}
@Override
public String toString() {
System.out.println("TeacherDao ClassLoader = " + this.getClass().getClassLoader() );
System.out.println(" Student ClassLoader = " + student.getClass().getClassLoader() );
return "TeacherDao [id=" + id + ", name=" + name + ", students="
+ student + "]";
}
}
StudentDao类代码:package com.bug315.dao;
public class StudentDao {
private int id;
private String name;
public StudentDao(int id, String name) {
this.id = id;
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Id=" + this.id + " Name=" + this.name;
}
}
测试类代码:package com.bug315.classloader;
import java.lang.reflect.Constructor;
import com.bug315.dao.StudentDao;
public class Test02 {
public static void main(String[] args) throws Exception {
// 引导类加载器
ClassLoader bootstrapLoader = String.class.getClassLoader();
System.out.println(bootstrapLoader);
// 系统类加载器
// 从ClassPath路径下面加载类
Class> studentClazz = Class.forName("com.bug315.dao.StudentDao");
System.out.println( studentClazz.getClassLoader() );
// 扩展类加载器
// 从jre/lib/ext目录下面加载类
Class> teacherClazz = Class.forName("com.bug315.dao.TeacherDao");
System.out.println( teacherClazz.getClassLoader() );
Object obj01 = teacherClazz.newInstance();
System.out.println( obj01 );
Constructor> constructor = teacherClazz.getConstructor(new Class[]{
int.class, String.class, StudentDao.class});
Object obj = constructor.newInstance(new Object[]{
10, "tom", new StudentDao(1, "zhangsan") });
System.out.println( obj );
}
}
测试结果:null
sun.misc.Launcher$AppClassLoader@ad3ba4
sun.misc.Launcher$ExtClassLoader@126b249
Exception in thread "main" java.lang.NoClassDefFoundError: com/bug315/dao/StudentDao
at java.lang.Class.getDeclaredConstructors0(Native Method)
at java.lang.Class.privateGetDeclaredConstructors(Class.java:2389)
at java.lang.Class.getConstructor0(Class.java:2699)
at java.lang.Class.newInstance0(Class.java:326)
at java.lang.Class.newInstance(Class.java:308)
at com.bug315.classloader.Test02.main(Test02.java:24)
Caused by: java.lang.ClassNotFoundException: com.bug315.dao.StudentDao
at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
at sun.misc.Launcher$ExtClassLoader.findClass(Launcher.java:229)
at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
... 6 more
从运行结果可以得出如下几点:
1、String等基础API的ClassLoader等于null,这就说明该类是通过引导类加载器进行加载的。
2、StudentDao类是通过系统类加载器(AppClassLoader)进行加载的,因为我将StudentDao放置到CLASSPATH环境变量下面。
3、TeacherDao类时通过扩展类加载器(ExtClassLoader)进行加载的,因为我将TeacherDao类放置到java_home/jre/lib/ext目录下面。
4、在创建TeacherDao类的对象时,抛出了ClassNotFoundException错误。因为TeacherDao类使用扩展类加载器进行加载的,因为扩展类加载器不加载Classpath下面的类。