来自:http://www.imooc.com/learn/372
一般情况下我们要在listview控件上展示自定义的数据,我们需要定义一个listview,然后对每个listview设置一个适配器adapter,一般继承自BaseAdapter,在每个适配器中我们都要定义一个viewholder,但是当我们有很多的listview时,这种做法就比较麻烦了,这时候我们就需要抽象出一个共同的适配器出来。
先看一个效果图,
1 新建一个Java bean
package com.baseadapter.bean;
/**
* 新建一个Java Bean类,用于描述listview中每一个item的信息
* 每一个item包括一个标题,一个描述信息,一个日期,一个电话
*/
public class Bean {
private String title;
private String desc;
private String time;
private String phone;
public Bean()
{
}
public Bean(String title, String desc, String time, String phone)
{
super();
this.title = title;
this.desc = desc;
this.time = time;
this.phone = phone;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getDesc() {
return desc;
}
public void setDesc(String desc) {
this.desc = desc;
}
public String getTime() {
return time;
}
public void setTime(String time) {
this.time = time;
}
public String getPhone() {
return phone;
}
public void setPhone(String phone) {
this.phone = phone;
}
}
2 在activity_main.xml中新建一个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/id_listview"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</RelativeLayout>
3 新建一个item布局文件,item.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:padding="10dp"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/id_title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:singleLine="true"
android:text="小白兔的故事1"
android:textColor="#444"/>
<TextView
android:id="@+id/id_desc"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textSize="14sp"
android:minLines="1"
android:maxLines="4"
android:layout_below="@id/id_title"
android:layout_marginTop="10dp"
android:text="第一天,小白兔去河边钓鱼,什么也没钓到,回家了。第二天,小白兔又去河边钓鱼,还是什么也没钓到,回家了。第三天,小白兔刚到河边,一条大鱼从河里跳出来,冲着小白兔大叫:你他妈的要是再敢用胡箩卜当鱼饵,我就扁死你!"
android:textColor="#898989"/>
<TextView
android:id="@+id/id_time"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="12sp"
android:layout_below="@id/id_desc"
android:layout_marginTop="10dp"
android:text="2015-12-31"
android:textColor="#898989"/>
<TextView
android:id="@+id/id_phone"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="12sp"
android:layout_below="@id/id_desc"
android:layout_marginTop="10dp"
android:text="10086"
android:drawableLeft="@drawable/mobile_phone"
android:drawablePadding="3dp"
android:padding="3dp"
android:layout_alignParentRight="true"
android:background="#2ed667"
android:textColor="#ffff"/>
</RelativeLayout>
现在我们使用传统的方式实现适配器的编写
1 新建MyAdapter继承自BaseAdapter
package com.example.imooc_baseadapter;
import java.util.List;
import javax.security.auth.PrivateCredentialPermission;
import com.baseadapter.bean.Bean;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
/**
* 新建一个listview的适配器,需要的参数有,Context,数据源即List<Bean>,
* 以及用LayoutInflater加载我们的item布局文件
*
*/
public class MyAdapter extends BaseAdapter {
private LayoutInflater mInflater;
private List<Bean> mDatas;
public MyAdapter(Context context, List<Bean> datas) {
mInflater = LayoutInflater.from(context);
mDatas = datas;
}
@Override
public int getCount() {
return mDatas.size();
}
@Override
public Object getItem(int position) {
return mDatas.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
//初始化我们的item中的控件
if (convertView == null) {
convertView = mInflater.inflate(R.layout.item, parent, false);//加载item布局文件
holder = new ViewHolder();
holder.mTitle = (TextView) convertView.findViewById(R.id.id_title);
holder.mDesc = (TextView) convertView.findViewById(R.id.id_desc);
holder.mTime = (TextView) convertView.findViewById(R.id.id_time);
holder.mPhone = (TextView) convertView.findViewById(R.id.id_phone);
convertView.setTag(holder);
}
else {
holder = (ViewHolder) convertView.getTag();
}
//为控件赋值
Bean bean = mDatas.get(position);
holder.mTitle.setText(bean.getTitle());
holder.mDesc.setText(bean.getDesc());
holder.mTime.setText(bean.getTime());
holder.mPhone.setText(bean.getPhone());
return convertView;
}
private class ViewHolder
{
//item中的四个控件
TextView mTitle;
TextView mDesc;
TextView mTime;
TextView mPhone;
}
}
2 MainActivity.java
package com.example.imooc_baseadapter;
import java.util.ArrayList;
import java.util.List;
import com.baseadapter.bean.Bean;
import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.Adapter;
import android.widget.ListView;
public class MainActivity extends Activity {
private ListView mListView;
private List<Bean> mDatas;//这里我们设置了多个item,因此需要存放4个Bean对象的list
private MyAdapter mAdapter;//声明自定义的适配器
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initDatas();//先调用这个方法,因为我们在这里给adapter设置了数据
initView();//在这里我们初始化了view,为listview设置了适配器
}
private void initView() {
mListView = (ListView) findViewById(R.id.id_listview);
mListView.setAdapter(mAdapter);
}
private void initDatas() {
//构造我们listview上的数据
mDatas = new ArrayList<Bean>();
Bean bean1 = new Bean("小白兔的故事1","第一天,小白兔去河边钓鱼,什么也没钓到,"
+ "回家了。第二天,小白兔又去河边钓鱼,还是什么也没钓到,回家了。第三天,小白兔刚到河边,"
+ "一条大鱼从河里跳出来,冲着小白兔大叫:你他妈的要是再敢用胡箩卜当鱼饵,我就扁死你!",
"2015-12-31","10086");
mDatas.add(bean1);
Bean bean2 = new Bean("小白兔的故事2","狼崽喜欢素食,狼爸妈很苦恼。一日,见狼崽狂追兔子,甚喜,"
+ "最终狼崽摁住兔子,恶狠狠地说:兔崽子,把胡萝卜交出来!",
"2015-12-31","10086");
mDatas.add(bean2);
Bean bean3 = new Bean("小白兔的故事3","一只青蛙偷亲了兔子一口撒腿就跑,兔子紧追不舍。青蛙情急之下跳进池塘,"
+ "不一会儿,一只癞蛤蟆爬了出来。兔子忍不住大笑:“哈哈,小样儿,皮肤过敏了吧!",
"2015-12-31","10086");
mDatas.add(bean3);
Bean bean4 = new Bean("小白兔的故事4","小白兔跑在大森林里,结果又迷路了,这时,它碰上一只小花兔,这回小白兔可学乖了,"
+ "跑过去说:“小花兔哥哥,小花兔哥哥,你要是告诉我怎样才能走出大森林,我就让你舒服舒服。”小花兔一听,"
+ "登时抡圆了给小白兔一个大嘴巴,说:“我靠,你丫是问路呐,还是找办呐?”",
"2015-12-31","10086");
mDatas.add(bean4);
Bean bean5 = new Bean("小白兔的故事5","蚂蚁在森林里走,突然遇到一只大象,蚂蚁连忙一头钻进土里,伸出一只腿。"
+ "小白兔见了很好奇,问:你在干什么?蚂蚁悄悄对它说:嘘……别出声,看我绊丫一跟头……",
"2015-12-31","10086");
mDatas.add(bean5);
mAdapter = new MyAdapter(this, mDatas);
}
}
优化—-编写通用的ViewHolder
我们对viewholder进行优化。建立一个通用的ViewHolder类,我们知道在ViewHolder中我们包含了对item中各种控件的引用,但是如果我们有多个listview的话,那他们的item中的控件不一定是相同的,因此,在通用的ViewHolder类中我们需要一个容器SparseArray(SparseArrays map integers to Objects)存储一个id及其对应的view控件,我们再新建一个getView(int id),通过id获取到我们的控件即可。
新建类ViewHolder.java
package com.baseadapter.utils;
import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* 构造一个通用的viewholder类
*
*/
public class ViewHolder {
private SparseArray<View> mViews;
private int mPosition;
private View mConvertView;
//layoutId即我们要引入的item布局文件的id
public ViewHolder(Context context, ViewGroup parent, int layoutId, int position)
{
this.mPosition = position;
this.mViews = new SparseArray<View>();
mConvertView = LayoutInflater.from(context).inflate(layoutId, parent, false);//引入我们的item布局文件
mConvertView.setTag(this);
}
//ViewHolder并不是每次都需要实例化,当convertview不为空时我们就不需要再实例化ViewHolder,因此我们增加一个入口方法
//来判断是否需要对ViewHolder实例化
public static ViewHolder get(Context context, View convertView, ViewGroup parent, int layoutId, int position)
{
if (convertView == null) {
return new ViewHolder(context, parent, layoutId, position);//返回一个实例化对象
}
else {
ViewHolder holder = (ViewHolder) convertView.getTag();
holder.mPosition = position;
return holder;
}
}
/**
* 通过viewId获取控件,此方法返回的是View的一个子类
*/
public <T extends View> T getView(int viewId)
{
View view = mViews.get(viewId);
if (view == null) {
//如果mViews没有相应的控件,我们就从convertView中找到这个控件,并将此控件和其id存放在mViews中
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
//mConvertView的get方法
public View getConvertView() {
return mConvertView;
}
}
新建一个相应的适配器MyAdapterWithCommonViewHolder
package com.example.imooc_baseadapter;
import java.util.List;
import javax.security.auth.PrivateCredentialPermission;
import com.baseadapter.bean.Bean;
import com.baseadapter.utils.ViewHolder;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
/**
* 新建一个listview的适配器,需要的参数有,Context,数据源即List<Bean>,
* 以及用LayoutInflater加载我们的item布局文件
*
*/
public class MyAdapterWithCommonViewHolder extends BaseAdapter {
private LayoutInflater mInflater;
private List<Bean> mDatas;
private Context mContext;
public MyAdapterWithCommonViewHolder(Context context, List<Bean> datas) {
this.mContext = context;
mInflater = LayoutInflater.from(context);
mDatas = datas;
}
@Override
public int getCount() {
return mDatas.size();
}
@Override
public Object getItem(int position) {
return mDatas.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//初始化ViewHolder
ViewHolder holder = ViewHolder.get(mContext, convertView, parent, R.layout.item, position);
//初始化我们的item中的控件
//为控件赋值
Bean bean = mDatas.get(position);
TextView title = holder.getView(R.id.id_title);
title.setText(bean.getTitle());
TextView desc = holder.getView(R.id.id_desc);
desc.setText(bean.getDesc());
TextView time = holder.getView(R.id.id_time);
time.setText(bean.getTime());
TextView phone = holder.getView(R.id.id_phone);
phone.setText(bean.getPhone());
//我们也可以写成((TextView)holder.getView(R.id.id_phone)).setText(bean.getPhone());
return holder.getConvertView();//返回convertView
}
}
然后在MainActivity中给我们的listview设置上面的适配器即可。
接下来,我们还可以继续优化。。见下一篇博客