在类加载过程的加载阶段,有一个通过类的全限定名获取描述类的二进制流的动作,Java虚拟机设计团队有意将这个动作的实现放到虚拟机之外实现,以便让用户自己决定如何获取所需类,这个动作的实现被称为类加载器。JVM根据职能的不同,设计了以下四种类加载器:
- 1、引导类加载器(BootStrap ClassLoader)
- 2、扩展类加载器(Extension ClassLoader)
- 3、应用程序类加载器(Application ClassLoader)
- 4、自定义类加载器(User ClassLoader)
前三种是虚拟机自带的类加载器,自定义类加载器是用户根据自己的需求设计的类加载器,下图是四种类加载器之间的关系,图中所表示的层次关系,并非类的继承关系,而是描述类加载器协作关系,通常这个协作关系通过组合的方式实现(设计模式中的组合,通过组合复用父类加载器的代码,除了引导类加载器,其他类加载器都有父类加载器),这个协作动作的实现被称为"双亲委派模型",后面的篇章中单独讲。
在Java程序中可按照如下代码获取除引导类加载器外的各个类加载器。
public class ClassLoaderTest { public static void main(String[] args) { // (启动类)系统类加载器: ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader(); System.out.println(systemClassLoader); //sun.misc.Launcher$AppClassLoader@18b4aac2 //扩展类加载器: ClassLoader extendClassLoader = systemClassLoader.getParent(); System.out.println(extendClassLoader); //sun.misc.Launcher$ExtClassLoader@1b6d3586 // 引导类加载器: ClassLoader bootstrapClassLoader = extendClassLoader.getParent(); System.out.println(bootstrapClassLoader); // null // 用户自定义的类默认用系统类加载器 ClassLoader classLoader = ClassLoaderTest.class.getClassLoader(); System.out.println(classLoader); // sun.misc.Launcher$AppClassLoader@18b4aac2 } }
一、引导类加载器
引导类加载器是由C/C++语言实现,嵌套在虚拟机内部,用来加载java核心库中的类,特性如下:
- 1、只加载JAVA_HOME/jre/lib/rt.jar、resources.jar或sun.boot.class.path路径下的内容
- 2、加载扩展类和应用程序类类加载器,并且是它们的父类加载器,不继承ClassLoader类,其本身没有父类加载器。
- 3、出于安全考虑,只加载包名为java、javax、sun开头的类,
- 4、引导类加载器无法通过getClassLoader()方法获取,只会返回null,如String.class.getClassLoader()。
二、扩展类加载器
扩展类加载器是Java语言实现的类加载器,在sun.misc.Launcher.ExtClassLoader中实现,派生于ClassLoader类,特性:
- 1、本身先有引导类加载器加载,父类加载器为引导类加载器(非继承关系)。
- 2、加载java.ext.dirs系统属性所指定的目录中的类库,或者加载JDK安装目录的jre/ext/dir扩展目录下的类库,用户可以将自己的jar放入该目录,通过扩展类加载器加载。
三、应用程序类加载器
应用程序类加载器也是Java语言实现的类加载器,在sun.misc.Launcher.AppClassLoader中实现,派生于ClassLoader类,特性:
- 1、本身先有引导类加载器加载,父类加载器为扩展类加载器(非继承关系)。
- 2、加载环境变量classpath或者java.class.path属性指定的目录下的类库。
- 3、程序中默认的类加载器,一般的Java应用程序都由它加载,可以通过java.lang.ClassLoader#getSystemClassLoader方法获取。
四、用户自定义类加载器
对于某一些特殊的类加载需求,用户可以通过继承ClassLoader实现自定义的类加载器,通过自定义类加载器,可以在以下的需求场景使用:
- 1、隔离类,如类路径冲突。
- 2、防反编译加密Class文件。
- 3、扩展类的加载源。
自定义实现类加载器可以通过继承java.lang.ClassLoader并重写loadClas()方法或者实现findClass()(推荐)方法,如下代码就是加载指定目录class的简单实现。
public class DirClassLoader extends ClassLoader { private String dir; // 指定目录 public DirClassLoader(String dir) { this.dir = dir; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { // class文件路径 String classPath = dir + File.separator + name + ".class"; try (InputStream is = new FileInputStream(classPath)){ byte[] b = new byte[is.available()]; is.read(b); return defineClass(name, b, 0, b.length); } catch (Exception ex) { throw new ClassNotFoundException(); } } public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException { DirClassLoader dirClassLoader = new DirClassLoader("D:\setup"); Object obj = dirClassLoader.loadClass("Hello").newInstance(); System.out.println(obj.getClass()); System.out.println(obj.getClass().getClassLoader()); } }
五、类在虚拟机中的唯一性
在虚拟机中,每个类的唯一性,由类和类加载器两个因素决定,也就是即使同一个类被不同类加载器加载,也会导致这两个类不相同。在上篇文章说过,类加载完成以后会在堆区生成一个类的java.lang.Class对象作为外部接口访问方法区对应的类信息,这个操作具象化就是Object的getClass()方法。getClass()获取到类的Class对象,从而根据getClassLoader()方法获取类加载器,下面代码中自定义了类加载器MyClassLoader,在main中通过自定义类加载器加载一次ClassLoaderUniqueTest类并创建实例,同时在执行main方法时,虚拟机会通过应用程序类加载器加载一次ClassLoaderUniqueTest类,最后通过instanceof比较类型,这个比较类型并不限于instanceof,还可以用equals。
public class ClassLoaderUniqueTest { // 自定义类加载器 static class MyClassLoader extends ClassLoader { @Override public Class<?> loadClass(String name) throws ClassNotFoundException { try { String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class"; InputStream is = this.getClass().getResourceAsStream(fileName); if (is == null) { return super.loadClass(name); } byte[] b = new byte[is.available()]; is.read(b); return defineClass(name, b, 0, b.length); } catch (IOException e) { throw new ClassNotFoundException(); } } } public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException { ClassLoader myClassLoader = new MyClassLoader(); Object obj = myClassLoader.loadClass("classloder.ClassLoaderUniqueTest").newInstance(); System.out.println(obj.getClass().getClassLoader()); System.out.println(obj instanceof classloder.ClassLoaderUniqueTest); } }
执行结果
classloder.ClassLoaderUniqueTest$MyClassLoader@74a14482
false