类加载中的属性问题

类加载中的属性问题

举三个例子来说明:

要准备的代码

package com.lhc.load;

public class SuperClass {
    static {
        System.out.println("SuperClass init");
    }
    public static int value = 123;
}


package com.lhc.load;

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

第一个例子

    @Test
    public void extendsTest(){
        System.out.println(SubClass.value);
    }
    
上述代码运行之后,只会输出“SuperClass init!",而不会输出“SubClass init!".对于静态字段,
只有直接定义这个字段的类才会被初始化,因此通过其子类来引用父类中定义的静态字段,只会触发父类
的初始化而不会触发子类的初始化。至于是否要触发子类的加载和验证,在虚拟机规范中并未明确规定,
这点取决于虚拟机的具体实现。对于Sun HotSpot虚拟机来说,可通过-XX:+TraceClassLoading参
数观察到此操作会导致子类的加载。
第二个例子

    @Test
    public void arrayTest(){
        SuperClass[] array =  new SuperClass[10];
    }
    
上述代码运行之后,没有输出“SuperClass init!",说明并没有触发类SuperClass的初始化阶段。
第三个例子

    @Test
    public void constTest(){
        System.out.println(ConstClass.value);
    }
    
上述代码运行之后,没有输出“ConstClass init!",这是因为虽然在Java源码中引用了ConstClass类中
的常量VALUE,但其实在编译阶段通过常量传播优化,已经将此常量的值"HelloWorld"存储到了测试主类的
常量池中,以后测试主类对常量ConstClass.VALUE的引用实际都被转化为测试主类对自身常量池的引用了。
也就是说,实际上测试主类的Class文件之中并没有CostClass类的符号引用入口,这两个类在编译成
Class之后就不存在任何联系了。

猜你喜欢

转载自blog.csdn.net/TP89757/article/details/103515990