编程语言
首页 > 编程语言> > java – 为什么我的字段为null,即使它应该立即实例化?

java – 为什么我的字段为null,即使它应该立即实例化?

作者:互联网

我声明并立即实例化的字段为空.这是一个示例代码:

public class NullFieldSSCCE {

    static abstract class Parent {
        List<String> values;
        Parent() {
            values = getValues();
        }

        protected abstract List<String> getValues();   
    }

    static class Child extends Parent {

        String param1="test1";
        String param2="test2";

        Child() {
        }

        @Override
        protected List<String> getValues() {
            return Arrays.asList( new String[] {param1, param2} );
        }
    }

    public static void main(String[] args) {
        Child child = new Child();

        System.out.println("Child p1="+child.values.get(0)+", p2="+child.values.get(1));
    }

}

运行它的结果是

Child p1=null, p2=null

虽然我期待

Child p1=test1, p2=test2

这怎么可能?这些字段在课程的同一时刻被实例化,不是吗?

解决方法:

怎么了

您遇到的内容将在JLS的8.3.2和8.8章中介绍.你可以找到一些不那么详细,也许更容易阅读的信息here.

这里的问题是对象的初始化顺序.

以下是您的示例的初始化顺序:

>父字段被实例化:值成为ArrayList的实例
>子构造函数调用super(),因此调用Parent构造函数
>调用getValues();在这种情况下,它是Child.getValues();
>子字段被实例化:param1和param2分配了它们的值
>子构造函数继续 – 在您的情况下不做任何其他事情

您可以保证给定层次结构中所有类的所有字段同时实例化,4在3之前发生,大致与1同时发生.遗憾的是,这种假设是错误的.

学习初始化顺序

即使您已经阅读了我给您的链接,您也可能无法确定在特定情况下的工作方式.每当您对初始化顺序有疑问时,请不要犹豫,添加printfs进行检查.我这次为你做了这件事:

public class NullFieldSSCCE {

    static abstract class Parent {
        List<String> values = new ArrayList<String>() {{
          System.out.println("Parent.values instantiation");  
        }};

        Parent() {
            System.out.println("Parent()");
            values.addAll(getValues());
        }

        protected abstract List<String> getValues();   
    }

    static class Child extends Parent {

        String param1="test1";
        String param2="test2";
        Object param3 = new Object() {{System.out.println("Child.param3 instantiation"); }};

        Child() {
            System.out.println("Child()");
        }

        @Override
        protected List<String> getValues() {
            System.out.println("Child.getValues()");
            return Arrays.asList( new String[] {param1, param2} );
        }
    }

    public static void main(String[] args) {
        System.out.println("start");
        Child child = new Child();

        System.out.println("Child p1="+child.values.get(0)+", p2="+child.values.get(1));
    }

}

输出是:

start
Parent.values instantiation
Parent()
Child.getValues()
Child.param3 instantiation
Child()
Child p1=null, p2=null

标签:java,field,instantiation,operator-precedence,class
来源: https://codeday.me/bug/20190612/1227452.html