Android打造listview万能适配器(上)

来自: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设置上面的适配器即可。

接下来,我们还可以继续优化。。见下一篇博客

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值