什么情况下,会触发类的初始化

本文解析了Java类初始化的时机,包括new、getstatic/putstatic/invokestatic、反射调用、主类执行和接口初始化规则。重点讲解了主动引用触发类初始化的情况,如静态变量、静态方法和MethodHandle。同时介绍了接口初始化的独特之处。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

引用

引用 《深入理解java虚拟机之jvm高级特性与最佳实践》的理解:

  1. 遇到new、getstatic 和 putstatic 或 invokestatic 这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。对应场景是:使用 new 实例化对象、读取或设置一个类的静态字段(被 final 修饰、已在编译期把结果放入常量池的静态字段除外)、以及调用一个类的静态方法。
  2. 对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
  3. 当初始化类的父类还没有进行过初始化,则需要先触发其父类的初始化。(而一个接口在初始化时,并不要求其父接口全部都完成了初始化)
  4. 虚拟机启动时,用户需要指定一个要执行的主类(包含 main() 方法的那个类),虚拟机会先初始化这个主类。
  5. 当使用 JDK 1.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果 REF_getStatic、REF_putStatic、REF_invokeStatic 的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。

第5种情况,暂时看不懂。

类初始化阶段进行的时机(主动引用)

  1. 创建类的实例;
  2. 访问类的静态变量;

注:当访问类的静态并且final修饰的变量时,不会触发类的初始化。或者为静态变量赋值。使用 new 关键字 赋值时,还是会触发类的初始化。

因为,常量在编译阶段会存入调用类的常量池中,常量是一种特殊的变量,编译器会把他们当作值(value)而不是域(field)来对待;如果你的代码中用到了常变量(constant variable),编译器并不会生成字节码来从对象中载入域的值,而是直接把这个值插入到字节码中。这是一种很有用的优化,但是如果你需要改变final域的值那么每一块用到那个域的代码都需要重新编译。

例如:

// 不会触发类的初始化
static final int count = 0;
// 触发类的初始化
static final Object obj = new Object();
  1. 调用类的静态方法;

注:调用静态且final的成员方法时,也会触发类的初始化!一定要和静态且final修饰的变量区分开!!

  1. 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化;

如:

Class.forName("com.xxx.Test");

注: 通过类名 .class 得到Class文件对象并不会触发类的加载。

  1. 当初始化一个类时,发现其父类还未初始化,则先出发父类的初始化

注:通过子类调用父类的静态成员时,只会初始化父类而不会初始化子类。 因为没有调用子类的相关静态成员,这也叫做能不加载不加载原则。

  1. 直接使用java.exe命令来运行某个主类;

注:java.exe运行,本质上就是调用main方法,所以必须要有main方法才行。

接口的初始化

接口的加载过程与类的加载过程稍有不同。接口中不能使用static{}块。当一个接口在初始化时,并不要求其父接口全部都完成了初始化,只有真正在使用到父接口时(例如引用接口中定义的常量)才会初始化

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值