谁能给我解释一下这个例子的行为:
package test;
public class Test {
static abstract class Parent{
public Parent() {
print();
}
public abstract void print();
}
static class Child extends Parent{
protected final int i= 10;
public Child() {
super();
}
public void print(){
System.out.println(i);
}
}
public static void main(String[] args) {
System.out.println("Test");
new Child();
}
}
此代码段的输出是 10。但是,当我将变量 i 更改为任何对象(例如 Integer)时,输出为 null。但是,当我将 i 更改为 static Integer 时,输出如预期的那样是 10。
我认为字段在调用任何方法(或构造函数)之前被初始化,但这里这种方法只适用于原始类型而不适用于对象。
谢谢
辐射
最佳答案
类的实例字段在构造函数中 super()
调用之后直接初始化,无论它们是否是最终的。但是,super()
调用父类的构造函数,它读取因此未初始化的字段的值。
类的静态字段在运行时引用类时被初始化,这意味着静态字段在类的构造函数被调用之前被初始化(在 super()
调用类的构造函数)。
当字段类型为 int 时,为什么仍然打印 10 的解释是,Java 编译器可能会在编译时内联最终字段的值。在这种情况下,程序永远不会读取该字段的值,因为编译器优化了代码,使其看起来像 System.out.println(10)
而不是 System.out.println(一)
。 Java 编译器在何种情况下内联 final 字段没有定义,但是,它很可能对原始字段类型这样做,而对大多数对象类型来说不太可能。
关于java - 最终字段的初始化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20523736/