JVM实践-类初始化

只有主动引用可以触发类的初始化,被动引用不能触发类的初始化。有以下5种主动引用:

1.遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果没有进行初始化,会触发初始化。

2.使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行初始化,则需触发初始化。

3.当初始化一个类时,如果发现其父类还没有进行初始化,则需先触发其父类的初始化。

4.虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个类。

5.当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄对应的类没有进行过初始化,则需先触发其初始化。

看一个被动引用的例子:

public class SuperClass {

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

    public static int value = 123;
}

 

public class SubClass extends SuperClass{

    static {
        System.out.println("SubClass init!");
    }
}

 

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

执行结果:

SuperClass init!
123

解释:对于静态字段,只有直接定义这个字段的类才会被初始化,因此通过其子类来引用父类中定义的静态字段,只会触发父类的初始化而不会触发子类的初始化。

 

常量在编译阶段会存入调用类的常量池中,以后对常量的引用实际上都被转换为对调用类自身常量池的引用,本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化:

public class ConstantClass {
    static {
        System.out.println("Constant init!");
    }
    public static final String HELLOWORD = "hello world";
}
public class NotInitialization {
    public static void main(String[] args){
        System.out.println(ConstantClass.HELLOWORD);
    }
}

结果:

hello world

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值