我们在前面三篇文章分别介绍了ArrayAdapter和SimpleAdapter的使用,可以先总结一下:
1)ArrayAdapter,是一个跟Array结构对应的Adapter,所以它展示的内容取决于Array里面装的对象,在默认或者大部分情况下,如果一个list只是要用来展示一些文字方面的效果,比如文章列表,联系人列表等比较简洁的描述,这是个最好的选择。当然,我们也可以通过继承它来自定义一个Adapter。
2)SimpleAdapter,需要1)我们自定义item的布局,2)需要我们将数据源封装成一个List>结构的列表中去,3)要将布局中的控件跟map中的数据对应起来。SimpleAdapter可以根据我们的需要,实现比较灵活的布局和效果,这方面是ArrayAdapter比不上的。
Android已经帮我们实现了这两个Adapter的逻辑,并封装得很好,好到我们只需要创建一个对象就可以用了。但有时候,包装得太过的东西总是麻烦,灵活性还是不够。
而在日常的开发中,我们经常会用到的,更多的是BaseAdapter,这个抽象类,而ArrayAdapter和SimpleAdapter,其实也是BaseAdapter的子类。
BaseAdapter
实现BaseAdapter,就比较麻烦一点,因为比较Base,所以需要我们实现的东西会比较多。
class MyAdapter extends BaseAdapter{
@Override
public int getCount() {
// TODO Auto-generated method stub
return 0;
}
@Override
public Object getItem(int position) {
// TODO Auto-generated method stub
return null;
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
// TODO Auto-generated method stub
return null;
}
}
首先我们要创建一个自定义的MyAdapter,并继承BaseAdapter,并且需要实现其几个基本的方法:
1)getCount():返回这个BaseAdapter处理数据源的总数,这个决定了下面会call多少次getView方法。
2)getItem():返回数据源中的某个位置的对象。
3)getItemId():返回数据源中的某个位置的对象的id。
4)getView():这个方法是整个Adapter方法中最重要的方法,它决定了我们在list上面展示的布局效果。
下面我们来实际应用一下吧。
我们还是采用上一篇文章中simpleadpater.xml的布局,如下:
然后我们来实现我们的Adapter,代码如下:
class MyAdapter extends BaseAdapter{
@Override
public int getCount() {
return titles.length;//返回titles.length,总共是6个。
}
@Override
public Object getItem(int position) {
// TODO Auto-generated method stub
return titles[position];
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//将布局inflate出来,然后获得布局上的控件,设置值
View view = LayoutInflater.from(MainActivity.this).inflate(R.layout.simpleadapter, null);
ImageView imageView = (ImageView) view.findViewById(R.id.imageView1);
imageView.setImageResource(drawableIds[position]);
TextView tvTitle = (TextView) view.findViewById(R.id.tvTitle);
tvTitle.setText("ba title " + position);
TextView tvContent = (TextView) view.findViewById(R.id.tvContent);
tvContent.setText("ba content" + position);
return view;
}
}
看效果图:
其实可以看到,效果跟simpleAdapter是一样的,那么如果我们返回10000个呢?我们改一下代码:
class MyAdapter extends BaseAdapter{
@Override
public int getCount() {
return 10000;
}
...
@Override
public View getView(int position, View convertView, ViewGroup parent) {
...
imageView.setImageResource(drawableIds[position % 6]);//取模,因为图片有限
...
}
}下面是效果图:
return createViewFromResource(position, convertView, parent, mResource);
}
private View createViewFromResource(int position, View convertView, ViewGroup parent,
int resource) {
View view;
TextView text;
if (convertView == null) {
view = mInflater.inflate(resource, parent, false);
} else {
view = convertView;
}
可以看到,会首先去判断convertView是否为null,当其为null的时候,才去重新创建一个view,不是的话,就直接用convertView。那么这个convertView是什么东西呢?其实它就是刚刚离开屏幕的那个View,所以我们可以复用它,这样我们就可以极大极大地减少这个view的创建。
所以我们把代码改成:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view ;
if(convertView == null){
view = LayoutInflater.from(MainActivity.this).inflate(R.layout.simpleadapter, null);
}else{
view = convertView;
}
...
}
这一点,可以说极大地优化了这个listView的性能,尤其是我们在加载图片这种很耗内存的资源的时候,这个效果更加明显。
不过这还不够,因为获得这个view之后,每次还需要通过 view.findViewById 来获得view里面的控件,findViewById的成本也是非常之高的,那么有什么办法呢?
一般来说,就是ViewHolder的应用了。
public View getView(int position, View convertView, ViewGroup parent) {
View view ;
ViewHolder viewHolder;
if(convertView == null){
view = LayoutInflater.from(MainActivity.this).inflate(R.layout.simpleadapter, null);
ImageView imageView = (ImageView) view.findViewById(R.id.imageView1);
TextView tvTitle = (TextView) view.findViewById(R.id.tvTitle);
TextView tvContent = (TextView) view.findViewById(R.id.tvContent);
viewHolder = new ViewHolder(imageView, tvTitle, tvContent);
view.setTag(viewHolder);
}else{
view = convertView;
viewHolder = (ViewHolder) view.getTag();
}
viewHolder.imageView.setImageResource(drawableIds[position % 6]);
viewHolder.tvTitle.setText("ba title " + position);
viewHolder.tvContent.setText("ba content" + position);
return view;
}
}
class ViewHolder {
public ViewHolder(ImageView imageView, TextView tvTitle, TextView tvContent){
this.imageView = imageView;
this.tvTitle = tvTitle;
this.tvContent = tvContent;
}
ImageView imageView;
TextView tvTitle;
TextView tvContent;
}
我们可以在从view中获得的这几个控件放到一个viewHolder的对象中,并将其设置给view的Tag属性,这样下一次再重复利用这个view的时候,就不必再通过findViewById来获得控件 ,而是可以直接通过viewHolder来获得对应的控件。
关于BaseAdapter的应用,基本上也就是这样。
关于ListView中几种Adapter的应用,几篇文章总算做了个总结,希望能够对大家有个帮助。
Android中关于Adapter的使用(上)ArrayAdapter
Android中关于Adapter的使用(再上)ArrayAdapter
Android中关于Adapter的使用(中)SimpleAdapter