java - 最终字段的初始化

标签 java initialization field

谁能给我解释一下这个例子的行为:

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/

相关文章:

java - Spring Boot RESTful 应用程序错误

java - 无法让正则表达式工作

c++ - 为什么初始化列表初始化需要堆分配数组的大小?

c++ - vector 的元素突然变得未初始化

ajax - 基于其他字段值的 Joomla 组件动态自定义字段

javascript - 使用 Dexie,我可以获取表中数组字段具有特定值作为其元素之一的所有对象吗?

java - Maven 似乎没有下载整个 jar

java - 在 Eclipse 中调试 Java 代码时出现 "Source not found"

c++ - 如何默认初始化 std::vector

module - 如何使用 Drupal 7 中的实体注册模块?