先看一段代码
@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中可以看出执行过程如下图:
注意:其中非静态变量和非静态代码块的执行顺序和其定义的先后有关。
明白了执行顺序后,我们回过来刚才的问题:
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。