4. 再谈类的加载器
4.1 概述
类加载器是JVM执行类加载机制的前提。
ClassLoader的作用:
ClassLoader是Java的核心组件,所有的Class都是由ClassLoader进行加载的,ClassLoader负责通过各种方式将Class信息的二进制数据流读入JVM内部,转换为一个与目标类对应的java.lang.Class对象实例。然后交给Java虚拟机进行链接、初始化等操作。因此,ClassLoader在整个装载阶段,只能影响到类的加载,而无法通过ClassLoader去改变类的链接和初始化行为。至于它是否可以运行,则由Execution Engine决定。
类加载器最早出现在Java 1.0 版本中,那个时候只是单纯地为了满足 Java Applet 应用而被研发出来。但如今类加载器却在OSGi、字节码加密领域大放异彩。这主要归功于Java虚拟机的设计者们当初在设计类加载器的时候,并没有考虑将他绑定在JVM内部,这样做的好处就是能够更加灵活地执行类加载操作。
4.1.1 大厂面试题
蚂蚁金服:
- 深入分析ClassLoader,双亲委派机制
- 类加载器的双亲委派模型是什么?一面:双亲委派机制及使用原因
百度:
- 都有哪些类加载器,这些类加载器都加载哪些文件?
- 手写一个类加载器Demo
- Class的forName(“java.lang.String”)和Class的getClassLoader()的Loadclass(“java.lang.String”)有什么区别?
腾讯:
- 什么是双亲委派模型?
- 类加载器有哪些?
小米:
- 双亲委派模型介绍一下
滴滴:
- 简单说说你了解的类加载器一面:讲一下双亲委派模型,以及其优点
字节跳动:
- 什么是类加载器,类加载器有哪些?
京东:
- 类加载器的双亲委派模型是什么?
- 双亲委派机制可以打破吗?为什么
4.1.2 类加载器的分类
类的加载分类:显式加载 vs 隐式加载
class文件的显式加载与隐式加载的方式是指JVM加载class文件到内存的方式。
- 显式加载指的是在代码中通过调用
ClassLoader
加载class对象,如直接使用Class.forName(name)
或this.getClass().getClassLoader().loadClass()
加载class对象。 - 隐式加载则是不直接在代码中调用
ClassLoader
的方法加载class对象,而是通过虚拟机自动加载到内存中,如在加载某个类的class文件时,该类的class文件中引用了另外一个类的对象,此时额外引用的类将通过JVM自动加载到内存中。
在日常开发以上两种方式一般会混合使用。
//隐式加载
User user=new User();
//显式加载,并初始化
Class clazz=Class.forName("com.test.java.User");
//显式加载,但不初始化
ClassLoader.getSystemClassLoader().loadClass("com.test.java.Parent");
4.1.3 类加载器的必要性
一般情况下,Java开发人员并不需要在程序中显式地使用类加载器,但是了解类加载器的加载机制却显得至关重要。从以下几个方面说:
- 避免在开发中遇到java.lang.ClassNotFoundException异常或java.lang.NoClassDefFoundError异常时,手足无措。只有了解类加载器的 加载机制才能够在出现异常的时候快速地根据错误异常日志定位问题和解决问题
- 需要支持类的动态加载或需要对编译后的字节码文件进行加解密操作时,就需要与类加载器打交道了。
- 开发人员可以在程序中编写自定义类加载器来重新定义类的加载规则,以便实现一些自定义的处理逻辑。
4.1.4 命名空间
何为类的唯一性?
对于任意一个类,都需要由加载它的类加载器和这个类本身一同确认其在Java虚拟机中的唯一性。每一个类加载器,都拥有一个独立的类名称空间:比较两个类是否相等,只有在这两个类是由同一个类加载器加载的前提下才有意义。否则,即使这两个类源自同一个Class文件,被同一个虚拟机加载,只要加载他们的类加载器不同,那这两个类就必定不相等。
命名空间
- 每个类加载器都有自己的命名空间,命名空间由该加载器及所有的父加载器所加载的类组成
- 在同一命名空间中,不会出现类的完整名字(包括类的包名)相同的两个类
- 在不同的命名空间中,有可能会出现类的完整名字(包括类的包名)相同的两个类
在大型应用中,我们往往借助这一特性,来运行同一个类的不同版本。
代码解释:
public static void main(String[] args) {
String rootDir ="D:\lcode\lworkspace_idea5\\]vMDemo1\\chapter04\\src\\";
try {
//创建自定义的类的加载器1
UserclassLoader loader1=new UserclassLoader(rootDir);
Class clazz1 =loader1.findclass( name: "com.atguigu.java.User"),
//创建自定义的类的加裁器2
UserClassLoader loader2 =new UserClassLoader(rootDir);
Class clazz2 =loader2.findclass( name:"com.atguigu.java.User");
System.out.println(clazz1 == clazz2); //clazz1与clazz2对应了不同的类模板结构。
System.out.println(clazz1.getclassLoader());
System.out.println(clazz2.getClassLoader());
//#####################
Class clazz3 = ClassLoader.getSystemclassLoader().loadclass( name: "com.atguigu.java.User");
System.out.println(clazz3.getclassLoader());
} catch (ClassNotFoundException e){
}
}
输出:
false
com.atguigu.java.UserclassLoader@1540e19d
com.atguigu.java.UserclassLoader@14ae5a5
sun.misc.Launcher$AppClassLoader@18b4aac2
解释:
rootDir后面的地址是我们使用javac User.class
指令生成的class文件地址,然后loader1
和loader2
是两个用户自定义类加载器(如果自定义的不必理解),之后使用这两个用户自定义类加载器加载同一类型的User
类,获得的Class对象不是同一个,可以通过Class对象调用getClassLoader()
方法获取对应的类加载器了,最后通过系统类加载器 获取的Class对象也是独特的,也可以通过该Class对象获取系统类加载器
4.1.5 类加载机制的基本特征
通常类加载机制有三个基本特征:
-
双亲委派模型。但不是所有类加载都遵守这个模型,有的时候,启动类加载器所加载的类型,是可能要加载用户代码的,比如JDK内部的ServiceProvider/ServiceLoader机制,用户可以在标准API框架上,提供自己的实现,JDK也需要提供些默认的参考实现。例如,Java中JNDI、JDBC、文件系统、Cipher等很多方面,都是利用的这种机制,这种情况就不会用双亲委派模型去加载,而是利用所谓的上下文加载器。
-
可见性,子类加载器可以访问父加载器加载的类型,但是反过来是不允许的。不然,因为缺少必要的隔离,我们就没有办法利用类加载器去实现容器的逻辑。
-
单一性,由于父加载器的类型对于子加载器是可见的,所以父加载器中加载过的类型,就不会在子加载器中重复加载。但是注意,类加载器“邻居”间,同一类型仍然可以被加载多次,因为互相并不可见。
4.1.6 类加载器之间的关系
public class Launcher {
// ……
public Launcher() {
Launcher.ExtClassLoader var1;
try {
var1 = Launcher.ExtClassLoader.getExtClassLoader();
} catch (IOException var10) {
throw new InternalError("Could not create extension class loader", var10);
}
try {
this.loader = Launcher.AppClassLoader.getAppClassLoader(var1);
} catch (IOException var9) {
throw new InternalError("Could not create application class loader", var9);
}
Thread.currentThread().setContextClassLoader(this.loader);
//……
}
}
TODO 笔记内容待补充
4.2 复习:类的加载器分类
JVM支持两种类型的类加载器,分别为引 导类加载器(Bootstrap ClassLoader)和自定义类加载器(User-Defined ClassLoader)。
从概念上来讲,自定义类加载器一般指的是程序中由开发人员自定义的一类类加载器,但是Java虚拟机规范却没有这么定义,而是将所有派生于抽象类ClassLoader的类加载器都划分为自定义类加载器。无论类加载器的类型如何划分,在程序中我们最常见的类加载器结构主要是如下情况:
- 除了顶层的启动类加载器外,其余的类加载器都应当有自己的“父类”加戟器。
- 不同类加载器看似是继承(Inheritance)关系,实际上是包含关系。在下层加载器中,包含着上层加载器的引用。
父类加载器和子类加载器的关系:
class ClassLoader{
ClassLoader parent;//父类加载器
public ClassLoader(ClassLoader parent){
this.parent = parent;
}
}
class ParentClassLoader extends ClassLoader{
public ParentClassLoader(ClassLoader parent){
super(parent);
}
}
class ChildClassLoader extends ClassLoader{
public ChildClassLoader(ClassLoader parent){ //parent = new ParentClassLoader();
super(parent);
}
}
正是由于子类加载器中包含着父类加载器的引用,所以可以通过子类加载器的方法获取对应的父类加载器
注意:
启动类加载器通过C/C++语言编写,而自定义类加载器都是由Java语言编写的,虽然扩展类加载器和应用程序类加载器是被jdk开发人员使用java语言来编写的,但是也是由java语言编写的,所以也被称为自定义类加载器
4.2.1 引导类加载器
启动类加载器(引导类加载器,Bootstrap ClassLoader)
- 这个类加载使用C/C++语言实现的,嵌套在JVM内部。
- 它用来加载Java的核心库(JAVAHOME/jre/lib/rt.jar或sun.boot.class.path路径下的内容)。用于提供JVM自身需要的类。
- 并不继承自java.lang.ClassLoader,没有父加载器。
- 出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类
- 加载扩展类和应用程序类加载器,并指定为他们的父类加载器。
使用-XX:+TraceClassLoading
参数得到。
启动类加载器使用C++编写的?Yes!
- C/C++:指针函数&函数指针、C++支持多继承、更加高效
- Java:由C++演变而来,(C++)--版,单继承
System.out.println("**********启动类加载器**********");
// 获取BootstrapclassLoader能够加载的api的路径
URL[] urLs = sun.misc.Launcher.getBootstrapClassPath().getURLs();
for (URL element : urLs) {
System.out.println(element.toExternalForm());
}
// 从上面的路径中随意选择一个类,来看看他的类加载器是什么:引导类加载器
ClassLoader classLoader = java.security.Provider.class.getClassLoader();
System.out.println(classLoader);
执行结果
**********启动类加载器**********
file:/D:/Java/jdk-1.8/jre/lib/resources.jar
file:/D:/Java/jdk-1.8/jre/lib/rt.jar
file:/D:/Java/jdk-1.8/jre/lib/jsse.jar
file:/D:/Java/jdk-1.8/jre/lib/jce.jar
file:/D:/Java/jdk-1.8/jre/lib/charsets.jar
file:/D:/Java/jdk-1.8/jre/lib/jfr.jar
file:/D:/Java/jdk-1.8/jre/classes
null
4.2.2 扩展类加载器
扩展类加载器(Extension ClassLoader)
- Java语言编写,由
sun.misc.Launcher$ExtClassLoader
实现。 - 继承于
ClassLoader
类 - 父类加载器为启动类加载器
- 从
java.ext.dirs
系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext
子目录下加载类库。如果用户创建的JAR放在此目录下,也会自动由扩展类加载器加载。
System.out.println("***********扩展类加载器***********");
String extDirs = System.getProperty("java.ext.dirs");
for (String path : extDirs.split(";")){
System.out.println(path);
}
// 从上面的路径中随意选择一个类,来看看他的类加载器是什么:扩展类加载器
ClassLoader classLoader1 = sun.security.ec.SunEC.class.getClassLoader();
System.out.println(classLoader1); //sun.misc. Launcher$ExtCLassLoader@1540e19d
执行结果:
***********扩展类加载器***********
D:\Java\jdk-1.8\jre\lib\ext
C:\Windows\Sun\Java\lib\ext
sun.misc.Launcher$ExtClassLoader@77459877
4.2.3 系统类加载器
应用程序类加载器(系统类加载器,AppClassLoader)
- java语言编写,由
sun.misc.Launcher$AppClassLoader
实现 - 继承于
ClassLoader
类 - 父类加载器为扩展类加载器
- 它负责加载环境变量classpath或系统属性
java.class.path
指定路径下的类库 - 应用程序中的类加载器默认是系统类加载器。
- 它是用户自定义类加载器的默认父加载器
- 通过
ClassLoader
的getSystemClassLoader()
方法可以获取到该类加载器
4.2.4 用户自定义类加载器
用户自定义类加载器
- 在Java的日常应用程序开发中,类的加载几乎是由上述3种类加载器相互配合执行的。在必要时,我们还可以自定义类加载器,来定制类的加载方式。
- 体现Java语言强大生命力和巨大魅力的关键因素之一便是,Java开发者可以自定义类加载器来实现类库的动态加载,加载源可以是本地的JAR包,也可以是网络上的远程资源。
- 通过类加载器可以实现非常绝妙的插件机制,这方面的实际应用案例举不胜举。例如,著名的OSGI组件框架,再如Eclipse的插件机制。类加载器为应用程序提供了一种动态增加新功能的机制,这种机制无须重新打包发布应用程序就能实现。
- 同时,自定义加载器能够实现应用隔离,例如Tomcat,Spring等中间件和组件框架都在内部实现了自定义的加载器,并通过自定义加载器隔离不同的组件模块。这种机制比C/C程序要好太多,想不修改C/C程序就能为其新增功能,几乎是不可能的,仅仅一个兼容性便能阻挡住所有美好的设想。
- 自定义类加载器通常需要继承于ClassLoader。
4.3 测试不同的类的加载器
每个Class对象都会包含一个定义它的ClassLoader的一个引用。
获取ClassLoader的途径
获得当前类的ClassLoader
clazz.getClassLoader()
获得当前线程上下文的ClassLoader
Thread.currentThread().getContextClassLoader()
获得系统的ClassLoader
ClassLoader.getSystemClassLoader()
说明:
- 站在程序的角度看,引导类加载器与另外两种类加载器(系统类加载器和扩展类加载器)并不是同一个层次意义上的加载器,引导类加载器是使用C++语言编写而成的,而另外两种类加载器则是使用Java语言编写而成的。由于引导类加载器压根儿就不是一个Java类,因此在Java程序中只能打印出空值。
- 数组类的Class对象,不是由类加载器去创建的,而是在Java运行期JVM根据需要自动创建的。对于数组类的类加载器来说,是通过Class.getClassLoader()返回的,与数组当中元素类型的类加载器是一样的;如果数组当中的元素类型是基本数据类型,数组类是没有类加载器的。
String[] strArr = new String[6];
System.out.println(strArr.getClass().getClassLoader());
// 运行结果:null
ClassLoaderTest[] test=new ClassLoaderTest[1];
System.out.println(test.getClass().getClassLoader());
// 运行结果:sun.misc.Launcher$AppCLassLoader@18b4aac2
int[]ints =new int[2];
System.out.println(ints.getClass().getClassLoader());
// 运行结果:null
获取当前线程上下文的ClassLoader的结果就是系统类加载器,这个可以在Launcher.java中被代码证明,即:
this.loader = Launcher.AppClassLoader.getAppClassLoader(var1);
Thread.currentThread().setContextClassLoader(this.loader);
public class ClassLoaderTest1{
public static void main(String[] args) {
//获取系统该类加载器
ClassLoader systemClassLoader=ClassLoader.getSystemCLassLoader();
System.out.print1n(systemClassLoader);//sun.misc.Launcher$AppCLassLoader@18b4aac2
//获取扩展类加载器
ClassLoader extClassLoader =systemClassLoader.getParent();
System.out.println(extClassLoader);//sun.misc. Launcher$ExtCLassLoader@1540e19d
//试图获取引导类加载器:失败
ClassLoader bootstrapClassLoader =extClassLoader.getParent();
System.out.print1n(bootstrapClassLoader);//null
//##################################
try{
ClassLoader classLoader =Class.forName("java.lang.String").getClassLoader();
System.out.println(classLoader);
//自定义的类默认使用系统类加载器
ClassLoader classLoader1=Class.forName("com.atguigu.java.ClassLoaderTest1").getClassLoader();
System.out.println(classLoader1);
//关于数组类型的加载:使用的类的加载器与数组元素的类的加载器相同
String[] arrstr = new String[10];
System.out.println(arrstr.getClass().getClassLoader());//null:表示使用的是引导类加载器
ClassLoaderTest1[] arr1 =new ClassLoaderTest1[10];
System.out.println(arr1.getClass().getClassLoader());//sun.misc. Launcher$AppcLassLoader@18b4aac2
int[] arr2 = new int[10];
System.out.println(arr2.getClass().getClassLoader());//null:
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
4.4 ClassLoader源码解析
除了以上虚拟机自带的加载器外,用户还可以定制自己的类加载器。Java提供了抽象类java.lang.ClassLoader,所有用户自定义的类加载器都应该继承ClassLoader类。
4.4.1 ClassLoader
的主要方法
抽象类ClassLoader的主要方法:(内部没有抽象方法)
public final ClassLoader getParent()
返回该类加载器的超类加载器
public Class<?> loadClass(String name) throws ClassNotFoundException