深入ClassLoader-类的加载过程

概述

ClassLoader的主要职责就是负责加载各种class文件到JVM中,ClassLoader是一个抽象类。给定一个clss的二进制名,ClassLoader会尝试加载并且在JVM中生成构成这个类的的各个数据结构,然后使其分布在JVM对应的内存区域。

类的加载过程简介

类的加载过程一般分为三个比较大的阶段,分别是加载阶段,连接阶段和初始化阶段。
在这里插入图片描述
加载阶段:主要负责查找并且加载类的二进制数据文件,其实就是class文件
连接阶段:连接阶段还可以分为三个阶段

  • 验证:主要是确保类文件的正确性,比如class的版本,class文件的魔术因子是否正确
  • 准备: 为类的静态变量分配内存,并且为其初始化默认值
  • 解析:把类中的符号引用转换为直接引用

初始化阶段:为类的静态变量赋予正确的初始值(代码编写阶段给定的值)
当一个JVM被我们启动时,包含的类非常多,但并不是每一个类都会被初始化。JVM对类的初始化是延迟机制,当一个类在首次主动使用时才会被初始化,在同一个运行时包下,一个Class只会被初始化一次

类的主动使用和被动使用

JVM虚拟机规范规定了,每个类或接口被Java程序首次主动使用时才会对其进行初始化。JVM同时规定了以下6种主动使用类的场景:

  • 通过new的关键字导致类的初始化
  • 访问类的静态变量导致类的初始化
  • 调用静态方法导致类的初始化
  • 对某个类进行反射操作导致类的初始化
  • 初始化子类会导致父类的初始化
  • 启动类:也就是执行main函数所在的类都会导致该类的初始化

除了上述6种情况,其余都称为被动使用,不会导致类的加载和初始化。下面几个栗子并不是主动使用,但是易混淆:

  • 构造某个类的数组并不会导致该类的初始化Simeple[] simples = new Simeple[10];不要被new所混淆,实际上只是开辟了10*Simple大小的内存
  • 引用类的常不会导致类的初始化,常量放在常量池中。
public class GlobleTest {
   
	
	static {
   
		System.out.println("类已被构造");
	}
	
	//在其他类中使用常量MAX不会导致GloableTest类被初始化,静态代码块不会执行
	public final static int MAX=100;
	//虽然random是静态常量,但是由于计算复杂,只有初始化之后才能得到结果
	//因此在其他类中调用random,会导致GlobleTest类被初始化
	public final static int random = new Random().nextInt();
}

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值