上一篇中我们介绍了自定义实现BaseAdapter的普通实现布局,然而上一章也说了普通实现的方式效率会很低,而且对系统开销也很大,所以,那样的实现是为了让初学者能知道可以这样使用,在实际项目中不可能使用那种方式的,要是你在做项目的时候使用普通布局方式,我敢保证,不过试用期你的老板就给你飞机票走人了,好了,闲话少说,本次讲解一下优化布局的实现,看完代码后,你会觉得,其实很简单。
MainActivity.java
public class MainActivity extends AppCompatActivity {
private List<Student> data;
private ListView mList;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mList = (ListView)findViewById(R.id.mList);
data = new ArrayList<>();
Student stu = null;
for(int i = 0; i < 5; i ++){
stu = new Student();
stu.setName("姓名"+ i);
stu.setSex(i % 2 == 0 ? "男" : "女");
data.add(stu);
}
MyAdapter adapter = new MyAdapter(data);
mList.setAdapter(adapter);
}
}
MyAdapter.java
public class MyAdapter extends BaseAdapter {
private List<Student> data;
public MyAdapter(List<Student> data) {
this.data = data;
}
@Override
public int getCount() {
return data == null ? 0 : data.size();
}
@Override
public Object getItem(int position) {
return data.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
/**
*
* @param position
* @param convertView
* @param parent
* @return
*/
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView == null){
//解析布局
convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item,null);
//创建ViewHolder持有类
holder = new ViewHolder();
//将每个控件的对象保存到持有类中
holder.tvName = (TextView)convertView.findViewById(R.id.mTv1);
holder.tvSex = (TextView)convertView.findViewById(R.id.mTv2);
//将每个convertView对象中设置这个持有类对象
convertView.setTag(holder);
}
//每次需要使用的时候都会拿到这个持有类
holder = (ViewHolder)convertView.getTag();
//然后可以直接使用这个类中的控件,对控件进行操作,而不用重复去findViewById了
holder.tvName.setText(data.get(position).getName());
holder.tvSex.setText(data.get(position).getSex());
return convertView;
}
/**
* 通过这个类来保存当前所有的控件id
*/
static class ViewHolder{
TextView tvName;
TextView tvSex;
}
}
上面的代码实现很简单,只是借助了一个ViewHolder持有类来保存每个布局中的控件ID就可以,也正如我们在上一篇普通实现BaseAdapter的第二种实现方式中所说的虽然使解析的次数降低到最少,但是每次都要findViewById,而这里的优化就刚好是针对那种方式的优化,这样一来,解析次数已经达到了最少,findViewById的次数也达到了最少,不过对于适配器来说,没有最优,只有更优,也就是说,如果以后我们在写ListView布局的的时候,每次都用这种方式去实现是非常方便,效率也不错,每写一个ListView处理不同的数据都要去自定义BaseAdapter,如果说一两个还好,要是你的项目中有20个处理不同数据的ListView,那么是不是要写20个自定义BaseAdapter呢?所以,为了方便开发,我们可以打造一个通用的BaseAdapter,关于如何打造一个通用的BaseAdapter,将在后面的学习笔记中介绍。