再看我们熟悉的ListAdapter

ListViewGirdView的数据和界面展示是通过调用setData(ListAdapter apapter)来实现。ListAdapter的子类有BaseAdapter,ArrayAdapter<T>,CursorAdapter,SimpleAdapter等等。如何选择合适的子类:

ArrayAdapter<T>

其典型构造方法是:

ArrayAdapter(Context context, int resource, int textViewResourceId, List<T> objects)

适合于展示单行文本内容,需指定布局资源文件resource,且该布局必须包含一个TextView,其id为textViewResourceId。显示的内容为TtoString(),无需重载getView方法。网上有不少错误的用法是使用复杂的布局结构且重载getView方法,将T的各个属性设置到对应布局控件中,这样显然违反了这个类的设计初衷。

SimpleAdapter

其构造方法是:

SimpleAdapter(Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to)

适用于显示含有若干TextviewImageView的布局。其缺点有:

1.构建数据集很麻烦,通常我们获取的数据为List<T>,T为业务实体类。这里需要的是Map<String,?>,需再转换一遍。

2.支持显示的view有限。TextView,ImageView和实现Checkable接口的view。可通过setViewBinder方法实现更多view绑定数据。但是自定义实现太多了,也违反了这个类的设计初衷。

CursorAdapter

其典型构造方法是:

CursorAdapter(Context context, Cursor c, boolean autoRequery)

该类的使用场景比较少,Cursor游标,通常是从数据库或content provider获取数据时才出现。个人建议,底层获取数据集游标,可预先遍历数据集,转换为List后再给上层UI使用。

BaseAdapter

该类比较灵活,是上面三个类的父类,也是我们最常使用的ListAdapter子类,基本使用流程是:

1.定义一个实体类

class TestItem {
    public String title;
    public String conent;
}

2.继承BaseAdapter

public class TestAdapter extends BaseAdapter {
    private List<TestItem> mData = new LinkedList<>();
    private Activity mContext;
    private LayoutInflater inflater;

    public TestAdapter(Activity context) {
        mContext = context;
        inflater = LayoutInflater.from(context);
    }

    public void setData(List<TestItem> data, boolean reset) {
        if (reset) {
            mData.clear();
        }
        if (data != null) {
            mData.addAll(data);
        }
        notifyDataSetChanged();
    }

    @Override
    public int getCount() {
        return mData.size();
    }

    @Override
    public TestItem getItem(int position) {
        return mData.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (convertView == null) {
            convertView = inflater.inflate(R.layout.refresh_list_item, null);
            holder = new ViewHolder();
            holder.titleTv = convertView.findViewById(R.id.title_tv);
            holder.contentTv = convertView.findViewById(R.id.content_tv);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }
        TestItem item = getItem(position);
        holder.titleTv.setText(item.title);
        holder.contentTv.setText(item.conent);
        return convertView;
    }

    private static class ViewHolder {
        public TextView titleTv;
        public TextView contentTv;
    }
}

在实现getView方法中,我们需要判断convertView是否已经可以复用,并且通过ViewHolder避免多次findViewById,而上面三个子类已经实现了view复用。仔细观察TestAdapter的实现,不难发现很多固定写法,能否进行一些抽象:

  1. 通过泛型代替具体实体类,setDatagetCountgetItemgetItemId可以固定实现

  2. 抽取更加通用的ViewHolder,支持任意view

  3. 绑定数据由具体业务实现

  4. 布局文件由具体业务决定

     public abstract class BaseAdapter<T> extends android.widget.BaseAdapter {
         protected Context mContext;
         protected List<T> mData = new LinkedList<>();
         protected LayoutInflater mInflater;
    
         public BaseAdapter(Activity context) {
             mContext = context;
             mInflater = LayoutInflater.from(context);
         }
    
         @Override
         final public int getCount() {
             return mData.size();
         }
    
         @Override
         final public T getItem(int position) {
             return mData.get(position);
         }
    
         /** 设置数据 */
         final public void setData(List<T> data, boolean reset) {
             if (reset) {
                 mData.clear();
             }
             if (data != null) {
                 mData.addAll(data);
             }
             notifyDataSetChanged();
         }
    
         @Override
         final public long getItemId(int position) {
             return position;
         }
    
         @Override
         final public View getView(int position, View convertView, ViewGroup parent) {
             int itemViewType = getItemViewType(position);
             T item = getItem(position);
             ViewHolder holder;
             boolean reused = false;
             if (convertView == null) {
                 int itemLayoutId = getItemLayoutId(itemViewType);
                 convertView = mInflater.inflate(itemLayoutId, null);
                 holder = new ViewHolder(convertView);
                 convertView.setTag(holder);
             } else {
                 reused = true;
                 holder = (ViewHolder) convertView.getTag();
             }
             handleItem(itemViewType, position, item, holder, reused);
             return convertView;
         }
    
         /**
          * 返回视图类型
          * @param itemViewType 视图类型
          * @return
          */
         protected abstract int getItemLayoutId(int itemViewType);
    
         /**
          * 处理item,主要是填充数据
          * @param itemViewType item对应的视图类型
          * @param position 实体对应的索引位置
          * @param item 具体实体
          * @param holder
          */
         protected abstract void handleItem(int itemViewType, int position, T item, ViewHolder holder, boolean reused);
    
         protected static class ViewHolder {
             private View mItemLayout;
             SparseArray<View> mViews = new SparseArray<View>();
    
             public ViewHolder(View itemLayout) {
                 mItemLayout = itemLayout;
             }
    
             public View gettemLayout() {
                 return mItemLayout;
             }
    
             public <T extends View> T get(int viewId) {
                 View childView = mViews.get(viewId);
                 if (childView == null) {
                     childView = mItemLayout.findViewById(viewId);
                     mViews.put(viewId, childView);
                 }
                 return (T) childView;
             }
    
             public <T extends View> T get(int viewId, Class<T> viewClass) {
                 View childView = mViews.get(viewId);
                 if (childView == null) {
                     childView = mItemLayout.findViewById(viewId);
                     mViews.put(viewId, childView);
                 }
                 return (T) childView;
             }
         }
    
     }
    

现在TestAdapter的实现可以很简单:

  private static class TestAdapter extends BaseAdapter<TestItem> {

        public TestAdapter(Activity context) {
            super(context);
        }

        @Override
        protected int getItemLayoutId(int itemViewType) {
            return R.layout.refresh_list_item;
        }

        @Override
        protected void handleItem(int itemViewType, int position, TestItem item, ViewHolder holder, boolean reused) {
            holder.get(R.id.title_tv, TextView.class).setText(item.title);
            holder.get(R.id.content_tv, TextView.class).setText(item.conent);
        }
    }

getItemLayoutId返回布局资源,当然可以根据itemViewType返回不同的布局。handleItem进行数据绑定。reused表明该view是否复用,通过holder的get(资源id)获取相应的控件。

结束

这一节介绍的ListAdapter是很基础的知识,大家应该非常熟悉,但是是否曾想抽象它,使我们的代码更加优雅!希望大家喜欢这个基础抽象类。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容