类加载中的属性问题
举三个例子来说明:
要准备的代码
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之后就不存在任何联系了。