Java成员变量和构造函数的执行顺序

先看一段代码

@NonNull
@Override
public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int itemType) {
    return new MyViewHolder(LayoutInflater.from(context).inflate(R.layout.item_fruit, parent, false));
}

public class MyViewHolder extends RecyclerView.ViewHolder {
    public ImageView icon = itemView.findViewById(R.id.icon);
    public TextView fruitName = itemView.findViewById(R.id.fruit_name);

    public MyViewHolder(@NonNull View itemView) {
        super(itemView);
    }
}

看了上段代码我有两个疑问:
public ImageView icon = itemView.findViewById(R.id.icon)在成员变量初始化时,直接使用itemView不为报空指针异常吗?
这个itemView是在调用new MyViewHolder(LayoutInflater.from(context).inflate(R.layout.item_fruit, parent, false));创建MyViewHolder时传入的View吗?
有这两个疑问说明对Java中父类和子类的成员变量和构造函数还不够了解,下面我们通过一个实例去看下它们的调用顺序到底如何。
Parent.java

public class Parent {
    public static String parentStaticString = "parent静态变量";
    protected String bb = "parent成员变量";

    static {
        System.out.println("parent:静态代码块:" + parentStaticString);
    }

    {
        System.out.println("parent:代码块");
    }

    public Parent() {
        System.out.println("parent:构造函数" + this.bb);

    }
}

Child.java

public class Child extends Parent {
   public static String childStaticString="child静态变量";

    static {
        System.out.println("child:静态代码块:"+childStaticString);
    }

    {
        System.out.println("child:代码块");
    }
    private String aa="child成员变量";

    public Child() {
        super();
        System.out.println("child:构造函数"+this.aa);
    }
}

通过Child child = new Child();来调用来看下代码执行的过程:

parent:静态代码块:parent静态变量
child:静态代码块:child静态变量
parent:代码块
parent:构造函数parent成员变量
child:代码块
child:构造函数child成员变量

从log中可以看出执行过程如下图:

执行顺序.png

注意:其中非静态变量和非静态代码块的执行顺序和其定义的先后有关。
明白了执行顺序后,我们回过来刚才的问题:

  • public ImageView icon = itemView.findViewById(R.id.icon)在成员变量初始化时,直接使用itemView不为报空指针异常吗?

当我们在调用new MyViewHolder(LayoutInflater.from(context).inflate(R.layout.item_fruit, parent, false));时会调用子类MyViewHolder的构造函数中的super(itemView),进而触发其父类RecyclerView.ViewHolder的构造函数,在父类的构造中将itemView赋值给父类成员变量的itemView
由于子类MyViewHolder的成员变量是在父类构造执行之后的,所以我们在这里调用itemView.findViewById(R.id.icon)时并不会为空。这个itemView也就是我们传入的View。

你可能感兴趣的:(Java成员变量和构造函数的执行顺序)