Main.java (总结在注释中)
public class Main {
public static void main(String[] args) {
/**
* REASON:
* 单纯的只初始化数组,不会触发相应的类实例化
*
* OUTPUT:
* empty output
*/
//SuperClazz[] sca = new SuperClazz[10];
/**
* REASON:
* 对于“静态”(static)字段 -- 直接引用 / 静态字段引用静态字段
*
* 只有直接定义这个字段的类才会被初始化,因此通过其子类来引用父类中定义的静态字段,只会触发父类的初始化而不会触发子类的初始化。
*
* OUTPUT:
* SuperClass init!
* 123
*/
//System.out.println(SubClazz.static_value);
//System.out.println(SubClazz.static_value_ref_static);
/**
* REASON:
* 对于“静态常量” (static final)字段 -- 直接引用 / 静态常量字段引用静态常量字段
*
* SuperClazz的“静态常量” (static final)字段 static_final_HELLOWORLD,其实在编译阶段通过“常量传播优化”,已经将SuperClazz的此常量值“hello world”存储到了SubClazz类的常量池中,
* 以后SubClazz对常量HELLOWORLD的引用实际都被转化为SubClazz类对自身常量池的引用了。
* 也就是说,实际上SubClazz的Class文件之中并没有SuperClazz类的符号引用入口,这两个类在编译成Class之后就不存在任何联系了.
*
* OUTPUT:
* hello,world
*/
//System.out.println(SubClazz.static_final_value);
//System.out.println(SubClazz.static_final_ref_static_final);
/**
* REASON:
* 对于混个状态下 -- 无论是“静态字段”引用“静态常量字段”, 还是“静态常量字段”引用“静态字段”
* 都需要初始化SuperClazz,不能使用“常量传播优化”。
* SubClazz的Class文件之中依然保存了SuperClazz类的符号引用入口。
*
* OUTPUT:
* SuperClass init!
* XXXXXXX(the real data)
*/
//System.out.println(SubClazz.static_ref_static_final);
//System.out.println(SubClazz.static_final_ref_static);
}
}
SuperClazz.java
public class SuperClazz {
static {
System.out.println("SuperClass init!");
}
public static int static_value = 123;
public static final String static_final_value = "hello,world";
public static int static_value_ref_static = static_value;
public static final String static_final_ref_static_final = static_final_value;
//-------------------
public static String static_value2 = "static_value_STRING";
public static final int static_final_value2 = 456;
public static int static_ref_static_final = static_final_value2;
public static final String static_final_ref_static = static_value2;
}
SubClazz.java
public class SubClazz extends SuperClazz {
static {
System.out.println("Subclass init!");
}
}