类加载器的分类
JVM支持两种类型的类加载器,分别为引导类加载器(Bootstrap ClassLoader)和自定义类加载器(User-Defined ClassLoader)。
从概念上来讲,自定义类加载器一般指的是程序中由开发人员自定义的一类类加载器,但是java虚拟机规范却没有这么定义,而是将所有派生于抽象类ClassLoader的类加载器都划分为自定义类加载器。
无论怎么划分,我们最常见的类加载器始终只有三种:
虚拟机自带的加载器
启动类加载器(引导类加载器 Bootstrap ClassLoader)
- 这个类加载使用C/C++语言实现,嵌套在JVM内部
- 它用来加载JAVA的核心库(JAVA_HOME/jre/lib/rt.jar,resources.jar或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类
- 并不继承自Java.lang.ClassLoader,没有父加载器
- 加载扩展类和应用程序类加载器,并指定为它们的父类加载器
- 出于安全考虑,Bootstrap启动类加载器只加载包名为java,javax,sun等开头的类
扩展类加载器(Extension ClassLoader)
- java语言编写,由sun.misc.Launcher$ExtClassLoader实现
- 派生于ClassLoader类
- 父类加载器为启动类加载器
- 从Java.ext.dirs系统属性所指的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录(扩展目录)下加载类库。如果用户创建的JAR放在此目录下,也会自动由扩展类加载器加载。
应用程序类加载器(系统类加载器 AppClassLoader)
- java语言编写,由sun.misc.Launcher&AppClassLoader实现
- 派生于ClassLoader类
- 父类加载器为扩展类加载器
- 它负责加载环境变量classpath或系统属性java.class.path指定路径下的类库
- 该类加载是程序中默认的类加载器,一般来说,Java应用的类都是由它来完成加载
- 通过ClassLoader#getSystemClassLoader()方法可以获得到该类加载器
用户自定义类加载器
在Java的日常应用程序开发中,类加载几乎是由上述3种类加载器相互配合执行的,在必要时,我们还可以自定义类加载器,来定制类的加载方式。
这里我们再提到一种十分常见的机制,叫双亲委派机制。
Java虚拟机对class文件采用的是按需加载的方式,也就是说当需要使用该类时才会将它的class文件加载到内存生成class对象,而且加载某个类的class文件时,Java虚拟机采用的是双亲委派模式,即把请求由父类处理,它是一种任务委派模式。
工作原理:
如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行;
如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归请求最终将到达顶层的启动类加载器;
如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子类加载器才会尝试自己去加载,这就是双亲委派模式。
我们再放一张示意图更加直观的了解双亲委派机制:
双亲委派机制的优势主要在于:
1.可以避免类的重复加载;
2.保护程序安全,防止核心API被随意篡改
由于我对JVM类加载器的学习还比较浅薄,有需要的话可以看一位知乎大佬对JVM更深的理解;
链接:https://zhuanlan.zhihu.com/p/185612299