-
1、概述
相信做Android开发的写得最多的就是ListView,GridView的适配器吧,记得以前开发一同事开发项目,一个项目下来基本就一直在写ListView的Adapter都快吐了~~~对于Adapter一般都继承BaseAdapter复写几个方法,getView里面使用ViewHolder模式,其实大部分的代码基本都是类似的。
本篇博客为快速开发系列的第一篇,将一步一步带您封装出一个通用的Adapter。
2、常见的例子
首先看一个最常见的案例,大家一目十行的扫一眼
1、布局文件
主布局文件:
12345<relativelayout android:layout_height="match_parent"android:layout_width="match_parent"xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><listview android:id="@+id/id_lv_main"android:layout_height="fill_parent"android:layout_width="fill_parent"></listview></relativelayout>
Item的布局文件:1234<!--?xml version=1.0encoding=utf-8?--><textview android:background="#aa111111"android:gravity="center_vertical"android:id="@+id/id_tv_title"android:layout_height="50dp"android:layout_width="match_parent"android:paddingleft="15dp"android:text="hello"android:textcolor="#ffffff"android:textsize="20sp"android:textstyle="bold"xmlns:android="http://schemas.android.com/apk/res/android"></textview>2、Adapter
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869packagecom.example.zhy_baseadapterhelper;importjava.util.List;importandroid.content.Context;importandroid.view.LayoutInflater;importandroid.view.View;importandroid.view.ViewGroup;importandroid.widget.BaseAdapter;importandroid.widget.TextView;publicclassMyAdapterextendsBaseAdapter{privateLayoutInflater mInflater;privateContext mContext;privateList<string> mDatas;publicMyAdapter(Context context, List<string> mDatas){mInflater = LayoutInflater.from(context);this.mContext = context;this.mDatas = mDatas;}@OverridepublicintgetCount(){returnmDatas.size();}@OverridepublicObject getItem(intposition){returnmDatas.get(position);}@OverridepubliclonggetItemId(intposition){returnposition;}@OverridepublicView getView(intposition, View convertView, ViewGroup parent){ViewHolder viewHolder =null;if(convertView ==null){convertView = mInflater.inflate(R.layout.item_single_str, parent,false);viewHolder =newViewHolder();viewHolder.mTextView = (TextView) convertView.findViewById(R.id.id_tv_title);convertView.setTag(viewHolder);}else{viewHolder = (ViewHolder) convertView.getTag();}viewHolder.mTextView.setText(mDatas.get(position));returnconvertView;}privatefinalclassViewHolder{TextView mTextView;}}</string></string>
3、Activity
123456789101112131415161718192021222324252627282930packagecom.example.zhy_baseadapterhelper;importjava.util.ArrayList;importjava.util.Arrays;importjava.util.List;importandroid.app.Activity;importandroid.os.Bundle;importandroid.widget.ListView;publicclassMainActivityextendsActivity{privateListView mListView;privateList<string> mDatas =newArrayList<string>(Arrays.asList(Hello,World, Welcome));privateMyAdapter mAdapter;@OverrideprotectedvoidonCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mListView = (ListView) findViewById(R.id.id_lv_main);mListView.setAdapter(mAdapter =newMyAdapter(this, mDatas));}}</string></string>
上面这个例子大家应该都写了无数遍了,MyAdapter集成BaseAdapter,然后getView里面使用ViewHolder模式;一般情况下,我们的写法是这样的:对于不同布局的ListView,我们会有一个对应的Adapter,在Adapter中又会有一个ViewHolder类来提高效率。这样出现ListView就会出现与之对于的Adapter类、ViewHolder类;那么有没有办法减少我们的编码呢?
下面首先拿ViewHolder开刀~
3、通用的ViewHolder
首先分析下ViewHolder的作用,通过convertView.setTag与convertView进行绑定,然后当convertView复用时,直接从与之对于的ViewHolder(getTag)中拿到convertView布局中的控件,省去了findViewById的时间~
也就是说,实际上们每个convertView会绑定一个ViewHolder对象,这个viewHolder主要用于帮convertView存储布局中的控件。
那么我们只要写出一个通用的ViewHolder,然后对于任意的convertView,提供一个对象让其setTag即可;
既然是通用,那么我们这个ViewHolder就不可能含有各种控件的成员变量了,因为每个Item的布局是不同的,最好的方式是什么呢?
提供一个容器,专门存每个Item布局中的所有控件,而且还要能够查找出来;既然需要查找,那么ListView肯定是不行了,需要一个键值对进行保存,键为控件的Id,值为控件的引用,相信大家立刻就能想到Map;但是我们不用Map,因为有更好的替代类,就是我们android提供的SparseArray这个类,和Map类似,但是比Map效率,不过键只能为Integer.
下面看我们的ViewHolder类:
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273packagecom.example.zhy_baseadapterhelper;importandroid.content.Context;importandroid.util.Log;importandroid.util.SparseArray;importandroid.view.LayoutInflater;importandroid.view.View;importandroid.view.ViewGroup;publicclassViewHolder{privatefinalSparseArray<view> mViews;privateView mConvertView;privateViewHolder(Context context, ViewGroup parent,intlayoutId,intposition){this.mViews =newSparseArray<view>();mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,false);//setTagmConvertView.setTag(this);}/*** 拿到一个ViewHolder对象* @param context* @param convertView* @param parent* @param layoutId* @param position* @return*/publicstaticViewHolder get(Context context, View convertView,ViewGroup parent,intlayoutId,intposition){if(convertView ==null){returnnewViewHolder(context, parent, layoutId, position);}return(ViewHolder) convertView.getTag();}/*** 通过控件的Id获取对于的控件,如果没有则加入views* @param viewId* @return*/public<textends=""view=""> T getView(intviewId){View view = mViews.get(viewId);if(view ==null){view = mConvertView.findViewById(viewId);mViews.put(viewId, view);}return(T) view;}publicView getConvertView(){returnmConvertView;}}</t></view></view>
与传统的ViewHolder不同,我们使用了一个SparseArray用于存储与之对于的convertView的所有的控件,当需要拿这些控件时,通过getView(id)进行获取;
下面看使用该ViewHolder的MyAdapter;
123456789101112@OverridepublicView getView(intposition, View convertView, ViewGroup parent){//实例化一个viewHolderViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,R.layout.item_single_str, position);//通过getView获取控件TextView tv = viewHolder.getView(R.id.id_tv_title);//使用tv.setText(mDatas.get(position));returnviewHolder.getConvertView();}
只看getView,其他方法都一样;首先调用ViewHolder的get方法,如果convertView为null,new一个ViewHolder实例,通过使用mInflater.inflate加载布局,然后new一个SparseArray用于存储View,最后setTag(this);如果存在那么直接getTag
最后通过getView(id)获取控件,如果存在则直接返回,否则调用findViewById,返回存储,返回。
好了,一个通用的ViewHolder写好了,以后一个项目几十个Adapter一个ViewHolder直接hold住全场~~大家可以省点时间斗个小地主了~~
4、打造通用的Adapter
有了通用的ViewHolder大家肯定不能满足,怎么也得省出dota的时间,人在塔在~~
下面看如何打造一个通过的Adapter,我们叫做CommonAdapter
继续分析,Adapter一般需要保持一个List对象,存储一个Bean的集合,不同的ListView,Bean肯定是不同的,这个CommonAdapter肯定需要支持泛型,内部维持一个List,就解决我们的问题了;
于是我们初步打造我们的CommonAdapter
我们的CommonAdapter依然是一个抽象类,除了getView以外我们把其他的代码都实现了,这样的话,在使用我们的Adapter只要实现一个getView,然后getView里面再使用我们打造的通过的ViewHolder是不是感觉还不错~1234567891011121314151617181920212223242526272829303132333435363738394041424344packagecom.example.zhy_baseadapterhelper;importjava.util.List;importandroid.content.Context;importandroid.view.LayoutInflater;importandroid.view.View;importandroid.view.ViewGroup;importandroid.widget.BaseAdapter;importandroid.widget.TextView;publicabstractclassCommonAdapter<t>extendsBaseAdapter{protectedLayoutInflater mInflater;protectedContext mContext;protectedList<t> mDatas;publicCommonAdapter(Context context, List<t> mDatas){mInflater = LayoutInflater.from(context);this.mContext = context;this.mDatas = mDatas;}@OverridepublicintgetCount(){returnmDatas.size();}@OverridepublicObject getItem(intposition){returnmDatas.get(position);}@OverridepubliclonggetItemId(intposition){returnposition;}}</t></t></t>现在我们的MyAdapter是这样的:
12345678910111213141516171819202122232425262728packagecom.example.zhy_baseadapterhelper;importjava.util.List;importandroid.content.Context;importandroid.view.View;importandroid.view.ViewGroup;importandroid.widget.TextView;publicclassMyAdapter<t>extendsCommonAdapter<t>{publicMyAdapter(Context context, List<t> mDatas){super(context, mDatas);}@OverridepublicView getView(intposition, View convertView, ViewGroup parent){ViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,R.layout.item_single_str, position);TextView mTitle = viewHolder.getView(R.id.id_tv_title);mTitle.setText((String) mDatas.get(position));returnviewHolder.getConvertView();}}</t></t></t>
所有的代码加起来也就10行左右,是不是神清气爽~~稍等,我先去dota一把~但是我们是否就这样满足了呢?显然还可以简化。
5、进一步铸造
注意我们的getView里面的代码,虽然只有4行,但是我觉得所有的Adapter的
第一行(ViewHolder viewHolder = getViewHolder(position, convertView,parent);)和
最后一行:return viewHolder.getConvertView();一定是一样的。
那么我们可以这样做:我们把第一行和最后一行写死,把中间变化的部分抽取出来,这不就是OO的设计原则嘛。现在CommonAdapter是这样的:
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364packagecom.example.zhy_baseadapterhelper;importjava.util.List;importandroid.content.Context;importandroid.view.LayoutInflater;importandroid.view.View;importandroid.view.ViewGroup;importandroid.widget.BaseAdapter;publicabstractclassn<t>extendsBaseAdapter{protectedLayoutInflater mInflater;protectedContext mContext;protectedList<t> mDatas;protectedfinalintmItemLayoutId;publicCommonAdapter(Context context, List<t> mDatas,intitemLayoutId){this.mContext = context;this.mInflater = LayoutInflater.from(mContext);this.mDatas = mDatas;this.mItemLayoutId = itemLayoutId;}@OverridepublicintgetCount(){returnmDatas.size();}@OverridepublicT getItem(intposition){returnmDatas.get(position);}@OverridepubliclonggetItemId(intposition){returnposition;}@OverridepublicView getView(intposition, View convertView, ViewGroup parent){finalViewHolder viewHolder = getViewHolder(position, convertView,parent);convert(viewHolder, getItem(position));returnviewHolder.getConvertView();}publicabstractvoidconvert(ViewHolder helper, T item);privateViewHolder getViewHolder(intposition, View convertView,ViewGroup parent){returnViewHolder.get(mContext, convertView, parent, mItemLayoutId,position);}}</t></t></t>
对外公布了一个convert方法,并且还把viewHolder和本Item对于的Bean对象给传出去,现在convert方法里面需要干嘛呢?通过ViewHolder把View找到,通过Item设置值;
现在我觉得代码简化到这样,我已经不需要单独写一个Adapter了,直接MainActivity匿名内部类走起~
123456789101112131415161718192021@OverrideprotectedvoidonCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mListView = (ListView) findViewById(R.id.id_lv_main);//设置适配器mListView.setAdapter(mAdapter =newCommonAdapter<string>(getApplicationContext(), mDatas, R.layout.item_single_str){@Overridepublicvoidconvert(ViewHolder c, String item){TextView view = viewHolder.getView(R.id.id_tv_title);view.setText(item);}});}</string>
可以看到效果咋样,不错吧。你觉得还能简化么?我觉得还能改善。6、Adapter最后的封魔
我们现在在convertView里面需要这样:
@Override
public void convert(ViewHolder viewHolder, String item)
{
TextView view = viewHolder.getView(R.id.id_tv_title);
view.setText(item);
}我们细想一下,其实布局里面的View常用也就那么几种:ImageView,TextView,Button,CheckBox等等;
那么我觉得ViewHolder还可以封装一些常用的方法,比如setText(id,String);setImageResource(viewId, resId);setImageBitmap(viewId, bitmap);
那么现在ViewHolder是:
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136packagecom.example.zhy_baseadapterhelper;importandroid.content.Context;importandroid.graphics.Bitmap;importandroid.util.SparseArray;importandroid.view.LayoutInflater;importandroid.view.View;importandroid.view.ViewGroup;importandroid.widget.ImageView;importandroid.widget.TextView;importcom.example.zhy_baseadapterhelper.ImageLoader.Type;publicclassViewHolder{privatefinalSparseArray<view> mViews;privateintmPosition;privateView mConvertView;privateViewHolder(Context context, ViewGroup parent,intlayoutId,intposition){this.mPosition = position;this.mViews =newSparseArray<view>();mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,false);// setTagmConvertView.setTag(this);}/*** 拿到一个ViewHolder对象** @param context* @param convertView* @param parent* @param layoutId* @param position* @return*/publicstaticViewHolder get(Context context, View convertView,ViewGroup parent,intlayoutId,intposition){if(convertView ==null){returnnewViewHolder(context, parent, layoutId, position);}return(ViewHolder) convertView.getTag();}publicView getConvertView(){returnmConvertView;}/*** 通过控件的Id获取对于的控件,如果没有则加入views** @param viewId* @return*/public<textends=""view=""> T getView(intviewId){View view = mViews.get(viewId);if(view ==null){view = mConvertView.findViewById(viewId);mViews.put(viewId, view);}return(T) view;}/*** 为TextView设置字符串** @param viewId* @param text* @return*/publicViewHolder setText(intviewId, String text){TextView view = getView(viewId);view.setText(text);returnthis;}/*** 为ImageView设置图片** @param viewId* @param drawableId* @return*/publicViewHolder setImageResource(intviewId,intdrawableId){ImageView view = getView(viewId);view.setImageResource(drawableId);returnthis;}/*** 为ImageView设置图片** @param viewId* @param drawableId* @return*/publicViewHolder setImageBitmap(intviewId, Bitmap bm){ImageView view = getView(viewId);view.setImageBitmap(bm);returnthis;}/*** 为ImageView设置图片** @param viewId* @param drawableId* @return*/publicViewHolder setImageByUrl(intviewId, String url){ImageLoader.getInstance(3, Type.LIFO).loadImage(url,(ImageView) getView(viewId));returnthis;}publicintgetPosition(){returnmPosition;}}</t></view></view>
现在的MainActivity只需要这么写:123456789mAdapter =newCommonAdapter<string>(getApplicationContext(),R.layout.item_single_str, mDatas){@Overrideprotectedvoidconvert(ViewHolder viewHolder, String item){viewHolder.setText(R.id.id_tv_title, item);}};</string>
convertView里面只要一行代码了~~~好了,到此我们的通用的Adapter已经一步一步铸造完毕~咋样,以后写项目省下来的时间是不是可以陪我切磋dota了(ps:11昵称:血魔哥404)~~
注:关于ViewHolder里面的setText,setImageResource这类的方法,大家可以在使用的过程中不断的完善,今天发现这个控件可以这么设置值,好,放进去;时间长了,基本就完善了。还有那个ImageLoader是我另一篇博客里的,大家可以使用UIL,Volley或者自己写个图片加载器;
7、实践
说了这么多,还是得拿出来让我们的实践检验检验,顺便来几张套图,俗话说,没图没正相。
1、我们的实例代码的图是这样的:

关于Adapter和ViewHolder的代码是这样的:
1234567891011// 设置适配器mListView.setAdapter(mAdapter =newCommonAdapter<string>(getApplicationContext(), mDatas, R.layout.item_single_str){@Overridepublicvoidconvert(ViewHolder helper, String item){helper.setText(R.id.id_tv_title,item);}});</string>
哎哟,我是不是只要贴一行;2、来个复杂点的布局
12345678910111213141516<!--?xml version=1.0encoding=utf-8?--><relativelayout android:background="#ffffff"android:layout_height="wrap_content"android:layout_width="match_parent"android:orientation="vertical"android:padding="10dp"xmlns:android="http://schemas.android.com/apk/res/android"><textview android:id="@+id/tv_title"android:layout_height="wrap_content"android:layout_width="match_parent"android:singleline="true"android:text="红色钱包"android:textcolor="#444444"android:textsize="16sp"></textview><textview android:id="@+id/tv_describe"android:layout_below="@id/tv_title"android:layout_height="wrap_content"android:layout_margintop="10dp"android:layout_width="match_parent"android:maxlines="2"android:minlines="1"android:text="周三早上丢失了红色钱包,在食堂二楼"android:textcolor="#898989"android:textsize="16sp"></textview><textview android:id="@+id/tv_time"android:layout_below="@id/tv_describe"android:layout_height="wrap_content"android:layout_margintop="10dp"android:layout_width="wrap_content"android:text="20130240122"android:textcolor="#898989"android:textsize="12sp"></textview><textview android:background="#5cbe6c"android:drawableleft="@drawable/icon_photo"android:drawablepadding="5dp"android:id="@+id/tv_phone"android:layout_alignparentright="true"android:layout_below="@id/tv_describe"android:layout_height="wrap_content"android:layout_margintop="10dp"android:layout_width="wrap_content"android:paddingbottom="3dp"android:paddingleft="5dp"android:paddingright="5dp"android:paddingtop="3dp"android:text="138024249542"android:textcolor="#ffffff"android:textsize="12sp"></textview></relativelayout>
效果图是这样的:
布局是不是挺复杂的了~~
但是代码是这样的:
12345678910111213141516// 设置适配器mListView.setAdapter(mAdapter =newCommonAdapter<bean>(getApplicationContext(), mDatas, R.layout.item_list){@Overridepublicvoidconvert(ViewHolder helper, Bean item){helper.setText(R.id.tv_title, item.getTitle());helper.setText(R.id.tv_describe, item.getDesc());helper.setText(R.id.tv_phone, item.getPhone());helper.setText(R.id.tv_time, item.getTime());// helper.getView(R.id.tv_title).setOnClickListener(l)}});</bean>
从一个字符串的布局到这样的布局,Adapter加ViewHolder的改变就这么多,加起来3行左右代码~~~到此,Android 快速开发系列 打造万能的ListView GridView 适配器结束;
最后给大家推荐一个gitHub项目:https://github.com/JoanZapata/base-adapter-helper ,这个项目所做的,和我上面写的基本一致。
还有上面的布局文件来自网络,感谢Bmob的提供~
好了,我要去快乐的玩耍了~~
Android 快速开发系列 打造万能的ListView GridView 适配器
最新推荐文章于 2024-04-06 06:05:29 发布
1179

被折叠的 条评论
为什么被折叠?



