JVM笔记 - 07类加载器深入解析及重要特性剖析


说明:笔记内容来源于《北京圣思园教育 - 张龙老师 - 深入理解JVM》视频课程。如有侵权,请联系删除。

1 类加载、连接、初始化的回顾

下图简单描述了类加载、连接、初始化、使用、卸载的顺序
在这里插入图片描述

下图相对详细的描述了类加载、连接、初始化、使用、卸载的顺序
在这里插入图片描述

2 类的加载

2.1 类的加载的最终产品是位于内存中的Class对象

2.2 Class对象封装了类在方法区内的数据结构

Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。
Class对象是整个反射的入口,Class对象就像一面镜子,能洞悉方法区内的所有信息,比如成员变量、方法等。

2.3 两种类型的类加载器

(1) Java虚拟机自带的加载器
① 根类加载器(Bootstrap)
② 扩展类加载器(Extension)
③ 系统(应用)类加载器(System)
(2)用户自定义的类加载器
特点:
① java.lang.ClassLoader的子类
② 用户可以定制类的加载方式

2.4 类加载器并不需要等到某个类被"首次主动使用"时再加载它

JVM笔记 - 02类的加载、连接、初始化过程详解-1.1 类的主动使用(七种) -⑤ 初始化一个类的子类
上面的链接中,演示了一个类没有被初始化,但已经被加载了。
详细解释:
a.JVM规范允许类加载器在预料某个类将要被使用时,就预先加载它。
b.如果在预先加载的过程中遇到了.class文件缺失或存在错误,
类加载器必须在程序首次主动使用该类时才报告错误(LinkageError错误)。
c.如果这个类一直没有被程序主动使用,那么类加载器就不会报告错误

3 类的连接

类被加载后,就进入连接阶段。
连接就是将已经被读到内存的类的二进制数据,合并到虚拟机的运行时环境中去。

3.1 类的验证的内容

(1) 类文件的结构检查
(2) 语义检查
(3) 字节码验证
(4) 二进制兼容性的验证

3.2 类的准备

在准备阶段,Java虚拟机为类的静态变量分配内存,并设置默认的初始值。
比如对于下面的Sample类,在准备阶段,将为 int 类型的静态变量 a 分配4个字节的内存空间,并且赋予默认值0;为long类型的静态变量 b 分配8个字节的内存空间,并且赋予默认值0。

package com.test;

public class Sample {
	private static int a = 1;
	public static long b;
	
	static {
		b =2;
	}
	//...
}

3.3类的初始化

在初始化阶段,Java虚拟机执行类的初始化语句,为类的静态变量赋予初始值。

3.3.1 静态变量的两种初始化途径

在程序中,静态变量的初始化有两种途径:
(1) 在静态变量的声明处进行初始化;
(2) 在静态代码块中进行初始化。
例如在以下代码中,静态变量 a 和 b 都被显示初始化,而静态变量 c 没有被显示初始化,它讲保持默认值0。

package com.test;

public class Sample {
	//在静态变量的声明处进行初始化
	private static int a = 1;
	public static long b;
	public static long c;
	
	static {
		//在静态代码块中进行初始化
		b =2;
	}
	//...
}

3.3.2 初始化语句

静态变量的声明语句,以及静态代码块都被看做类的初始化语句。
Java虚拟机会按照初始化语句在类文件中的先后顺序来依次执行它们。
例如当以下Sample类被初始化后,它的静态变量 a 的取值为4。

package com.test;

public class Sample {
	static int a = 1;
	
	static {
		a =2;
	}
	
	static {
		a =4;
	}
	
	public static void main(String[] args) {
		System.out.println("a = " + a);
	}
}

在这里插入图片描述

3.3.3 类的初始化步骤

类的初始化步骤:
(1) 假如这个类还没有被加载和连接,那就先进行加载和连接
(2) 假如类存在直接父类,并且这个父类还没有被初始化,那就先初始化直接父类
(3) 假如类中存在初始化语句,那就依次执行这些初始化语句

3.3.4 类的初始化时机

(1) 类的主动使用(七种,重要)
① 创建类的实例
② 访问某个类或接口的静态变量,或者对该静态变量赋值
③ 调用类的静态方法
④ 反射(如Class.forName(“com.test.Test”))
⑤ 初始化一个类的子类
⑥ Java虚拟机启动时被标明为启动类的类
⑦ JDK1.7开始提供的动态语言支持:
java.lang.invoke.MethodHandle实例的解析结果REF_getStatic,REF_putStatic,REF_invokeStatic句柄对应的类没有初始化,则初始化。

除了上述七种情形,其他使用Java类的方式,都被看作是被动使用,不会导致类的初始化。

(2) 父子类、父子接口的初始化
当Java虚拟机初始化一个类时,要求它的所有父类都已经被初始化,但是这条规则并不适用于接口。
① 在初始化一个类时,并不会先初始化它所实现的接口
② 在初始化一个接口时,并不会先初始化它的父接口
因此,一个父接口并不会因为它的子接口 或者实现类的初始化而初始化。
只有当程序首次使用特定接口的静态变量时,才会导致该接口的初始化。
JVM笔记 - 05接口的初始化规则中有具体案例的展示。

(3) 静态变量 和主动使用
只有当程序访问的静态变量 或静态方法确实在当前类 或当前接口中定义时,才可以认为是对类或接口的主动使用。

(4) 调用ClassLoader类的loadClass方法加载一个类,并不是对类的主动使用,不会导致类的初始化。

package com.test;

public class LoaderTest {

	public static void main(String[] args) throws ClassNotFoundException {
		ClassLoader loader = ClassLoader.getSystemClassLoader();
		Class<?> clazz= loader.loadClass("com.test.Test");
		System.out.println(clazz);
		
		System.out.println("-------------------------");
		clazz = Class.forName("com.test.Test");
		System.out.println(clazz);
	}
}

class Test {
	static {
		System.out.println("Test loading...");
	}
}

执行结果为:
在这里插入图片描述

4. 类加载器

4.1类加载器简介

类加载器用来把类加载到Java虚拟机中。
从JDK1.2版本开始,类的加载过程采用父亲委托机制,这种机制能更好地保证Java平台的安全。
在父亲委托机制中,
除了Java虚拟机自带的根类加载器以外,其余的类加载器都有且只有一个父加载器。
当Java程序请求加载器loader1加载Sample类时,loader1首先委托自己的父加载器去加载Sample类,
若父加载器能加载,则由父加载器完成加载任务,
否则才由加载器loader1本身加载Sample类。

4.2 类加载器的类型

Java虚拟机自带了3种加载器
(1) 根(Bootstrap)类加载器
根类加载器没有父加载器。
根类加载器负责加载虚拟机的核心类库,如java.lang.*等。
根类加载器从系统属性sun.boot.class.path所指定的目录中加载类库。
根类加载器的实现依赖于底层操作系统,属于虚拟机的实现的一部分,它并没有继承java.lang.ClassLoader类。

(2) 扩展(Extension)类加载器
扩展类加载器的父加载器为根类加载器。
扩展类加载器从java.ext.dirs系统属性所指定的目录中加载类库,或者从JDK的安装目录的jre\lib\ext子目录(扩展目录)下加载类库。
如果把用户创建的jar文件放在jre\lib\ext目录下,也会自动由扩展类加载器加载。
扩展类加载器是纯Java类,是java.lang.Classloader类的子类。

(3) 系统(System)类加载器
系统类加载器 也称为应用类加载器。
系统类加载器的父加载器为扩展类加载器。
系统类加载器从环境变量classpath或者系统属性java.class.path所指定的目录中加载类。
系统类加载器是用户自定义的类加载器的默认父加载器。
系统类加载器是纯Java类,是java.lang.ClassLoader类的子类。

(4) 自定义的类加载器
用户可以定制自己的类加载器。
Java提供了抽象类java.lang.ClassLoader,所有用户自定义的类加载器都应该继承ClassLoader类。

4.3 类加载器的包含关系

在这里插入图片描述
系统类加载器 的父加载器是 扩展类加载器,
扩展类加载器 的父加载器是 根类加载器。

系统类加载器在加载一个类Sample时,先由扩展类加载器去加载Sample,
扩展类加载器在加载Sample时,先由根类加载器去加载Sample。
根类加载器不能加载Sample,再由扩展类加载器去加载Sample,
扩展类加载器不能加载Sample,再由系统类加载器去加载Sample。

所以,系统类加载器包含了 扩展类加载器,
扩展类加载器包含了根类加载器。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值