类初始化时机

对于类加载过程中初始化阶段,虚拟机严格规定了有且只有四种情况必须立即对类进行“初始化”:

  1. 遇到new、getstatic、putstatic和invokestatic这4条指令码时,如果类没有初始化,则需要先进行初始化。这4条指令对应的java场景是:
    1. 使用new关键字实例化对象
    2. 读取或设置一个静态字段,不包括被final修饰的、已在编译器把结果放入常量池的静态字段
    3. 调用一个静态方法
  2. 使用java.lang.reflect包方法对类进行发射调用
  3. 当初始化一个类的时候,发现其父类还没有进行过初始化,则需要先初始化其父类。但是对于一个接口来说,只有在用到其父接口的时候才会去初始化。
  4. 当虚拟机启动时,用户需要指定一个要执行的类,虚拟机会先初始化这个类

以上这四种情况被称为是对一个类的主动引用。除此之外所有引用类的方式,都不会触发初始化,被称为被动应用。

下面是几个被动引用的例子,假设

class SuperClass{
    public static int value = 13;

    static {
        System.out.println("SuperClass Init");
    }
}

class SubClass extends SuperClass{
    static {
        System.out.println("SubClass Init");
    }
}


  1. 通过子类引用父类的静态字段,不会导致子类初始化

public class ClassLoadTest {
    public static void main(String[] args) {
        System.out.println(SubClass.value);
    }
}


#=>SuperClass Init

#=>13

对于静态字段,只有直接定义这个字段的类才会被初始化。

  1. 通过数组来定义引用类,不会触发此类的实例化

public class ClassLoadTest {
    public static void main(String[] args) {
        SuperClass[] superClasses = new SuperClass[10];
    }
}


#=>

  1. 常量在编译阶段会存入调用类的常量池,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。

class SuperClass{
    public static final int value = 13;

    static {
        System.out.println("SuperClass Init");
    }
}

public class ClassLoadTest {
    public static void main(String[] args) {
        System.out.print(SuperClass.value);
    }
}

#=>

在编译阶段SuperClass.value的值就被存在了ClassLoadTest的常量池中了,对SuperClass.value的引用实际上相当于ClassLoadTest内部的引用,所以在ClassLoadTest的类文件中并没有SuperClass类的符号引用入口,这两个类在编译后没有任何关系了。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值