在我们使用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