在我们使用ListView的时候,大部分时间我们使用的是ArrayAdapter或者SimpleAdapter,但是当ListView的列表项中出现按钮,复选框等附带事件的控件时,数据是无法映射到这些控件的,即使采用自定义布局文件显示控件,它们的事件监听也不会响应,这时就需要使用BaseAdapter自定义适配器,ArrayAdapter和SimpleAdapter都是直接继承于BaseAdapter
自定义适配器需要建立一个新类继承自BaseAdapter,并且覆写该类中的四个抽象方法
public int getCount(); 获取item的总数
public Object getItem(int position); 获取自定索引对应的数据项
public long getItemId(int position); 获取对应项的id,也就是position
public View getView(int position, View convertView, ViewGroup parent); 获取对应item的view,用于显示
我们在一个实例中实现自定义适配器
第一步:定义主ActivityTestListView局文件 为了简单期间主布局文件仅仅包含一个ListView
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<ListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="match_parent" >
</ListView>
</RelativeLayout>
第二步:定义ListView每一项的布局文件,包含一张图片,以及两个TextView
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<ImageView
android:id="@+id/img"
android:layout_width="60dp"
android:layout_height="60dp"
android:src="@drawable/ic_launcher" />
<TextView
android:id="@+id/title"
android:layout_width="wrap_content"
android:layout_height="30dp"
android:layout_toRightOf="@+id/img"
android:gravity="center_vertical"
android:text="标题" />
<TextView
android:id="@+id/content"
android:layout_width="wrap_content"
android:layout_height="30dp"
android:layout_below="@id/title"
android:layout_toRightOf="@+id/img"
android:gravity="center_vertical"
android:text="内容" />
</RelativeLayout>
第三步:准备数据源,使用List来保存需要显示的数据,为了方便封装,先准备一个工具类,用于对应每一个item的数据
public class ItemBean {
private int itemImg;
private String itemTitle;
private String itemContent;
public ItemBean(int itemImg, String itemTitle, String itemContent) {
this.itemImg = itemImg;
this.itemTitle = itemTitle;
this.itemContent = itemContent;
}
public int getItemImg() {
return itemImg;
}
public void setItemImg(int itemImg) {
this.itemImg = itemImg;
}
public String getItemTitle() {
return itemTitle;
}
public void setItemTitle(String itemTitle) {
this.itemTitle = itemTitle;
}
public String getItemContent() {
return itemContent;
}
public void setItemContent(String itemContent) {
this.itemContent = itemContent;
}
}
第四步:最重要的一步,自定义MyAdapter继承自BaseAdapter,并且覆写其四个抽象方法,定义一个构造函数,用于传递数据,由于显示View需要Context对象,所以构造函数的参数有两个
public class MyAdapter extends BaseAdapter {
private Context context;
private List<ItemBean> data;
/**
* 构造函数,ItemBean为自定义的工具类,用于保存每一项的数据
*/
public MyAdapter(Context context, List<ItemBean> data) {
this.context = context;
this.data = data;
}
// 获取listview的总数
@Override
public int getCount() {
return data.size();
}
// 获取与指定索引相对应的数据集
// 由于前面每一项的数据已经封装成ItemBean类了,所以直接返回这个对象即可
@Override
public Object getItem(int position) {
return data.get(position);
}
// 获取listview对应项的id
@Override
public long getItemId(int position) {
return position;
}
// 获取item显示的内容
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//获取对应itme的数据
ItemBean item = data.get(position);
View view = View.inflate(context, R.layout.lv_item, null);
ImageView img = (ImageView) view.findViewById(R.id.img);
TextView title = (TextView) view.findViewById(R.id.title);
TextView content = (TextView) view.findViewById(R.id.content);
//为对应item设置数据
img.setImageResource(item.getItemImg());
title.setText(item.getItemTitle());
content.setText(item.getItemContent());
//返回设置好的view,用于显示
return view;
}
}
最后一步:前面已经准备好了数据源,还有自定义好了适配器,现在可以在Activity中使用了
public class MainActivity extends ActionBarActivity {
private ListView listview;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.fragment_main);
listview = (ListView) findViewById(R.id.listView);
MyAdapter adapter = new MyAdapter(this, getData());
listview.setAdapter(adapter);
}
private List<ItemBean> getData() {
List<ItemBean> data = new ArrayList<ItemBean>();
for (int i = 0; i < 11; i++) {
ItemBean item = new ItemBean(R.drawable.ic_launcher, "标题" + i, "内容" + i);
data.add(item);
}
return data;
}
}
效果图:(BaseAdapter中getView的这种实现是不优雅的,不高效的,优雅的使用下一篇文章再详解)
源码下载:360云盘
http://yunpan.cn/cuUvgEN96mP9H 访问密码 c6fe
