多多色-多人伦交性欧美在线观看-多人伦精品一区二区三区视频-多色视频-免费黄色视屏网站-免费黄色在线

國內(nèi)最全I(xiàn)T社區(qū)平臺(tái) 聯(lián)系我們 | 收藏本站
阿里云優(yōu)惠2
您當(dāng)前位置:首頁 > 互聯(lián)網(wǎng) > Android 快速開發(fā)系列 打造萬能的ListView GridView 適配器

Android 快速開發(fā)系列 打造萬能的ListView GridView 適配器

來源:程序員人生   發(fā)布時(shí)間:2014-09-09 17:30:45 閱讀次數(shù):2810次

轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/lmj623565791/article/details/38902805 ,本文出自【張鴻洋的博客】

1、概述

相信做Android開發(fā)的寫得最多的就是ListView,GridView的適配器吧,記得以前開發(fā)一同事開發(fā)項(xiàng)目,一個(gè)項(xiàng)目下來基本就一直在寫ListView的Adapter都快吐了~~~對(duì)于Adapter一般都繼承BaseAdapter復(fù)寫幾個(gè)方法,getView里面使用ViewHolder模式,其實(shí)大部分的代碼基本都是類似的。

本篇博客為快速開發(fā)系列的第一篇,將一步一步帶您封裝出一個(gè)通用的Adapter。

2、常見的例子

首先看一個(gè)最常見的案例,大家一目十行的掃一眼

1、布局文件

主布局文件:

<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_lv_main" android:layout_width="fill_parent" android:layout_height="fill_parent" /> </RelativeLayout>

Item的布局文件:

<?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/id_tv_title" android:layout_width="match_parent" android:layout_height="50dp" android:background="#aa111111" android:gravity="center_vertical" android:paddingLeft="15dp" android:textColor="#ffffff" android:text="hello" android:textSize="20sp" android:textStyle="bold" > </TextView>

2、Adapter

package com.example.zhy_baseadapterhelper; import java.util.List; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.TextView; public class MyAdapter extends BaseAdapter { private LayoutInflater mInflater; private Context mContext; private List<String> mDatas; public MyAdapter(Context context, List<String> mDatas) { mInflater = LayoutInflater.from(context); this.mContext = context; this.mDatas = mDatas; } @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 = null; if (convertView == null) { convertView = mInflater.inflate(R.layout.item_single_str, parent, false); viewHolder = new ViewHolder(); viewHolder.mTextView = (TextView) convertView .findViewById(R.id.id_tv_title); convertView.setTag(viewHolder); } else { viewHolder = (ViewHolder) convertView.getTag(); } viewHolder.mTextView.setText(mDatas.get(position)); return convertView; } private final class ViewHolder { TextView mTextView; } }

3、Activity

package com.example.zhy_baseadapterhelper; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import android.app.Activity; import android.os.Bundle; import android.widget.ListView; public class MainActivity extends Activity { private ListView mListView; private List<String> mDatas = new ArrayList<String>(Arrays.asList("Hello", "World", "Welcome")); private MyAdapter mAdapter; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mListView = (ListView) findViewById(R.id.id_lv_main); mListView.setAdapter(mAdapter = new MyAdapter(this, mDatas)); } }

上面這個(gè)例子大家應(yīng)該都寫了無數(shù)遍了,MyAdapter集成BaseAdapter,然后getView里面使用ViewHolder模式;一般情況下,我們的寫法是這樣的:對(duì)于不同布局的ListView,我們會(huì)有一個(gè)對(duì)應(yīng)的Adapter,在Adapter中又會(huì)有一個(gè)ViewHolder類來提高效率。

這樣出現(xiàn)ListView就會(huì)出現(xiàn)與之對(duì)于的Adapter類、ViewHolder類;那么有沒有辦法減少我們的編碼呢?

下面首先拿ViewHolder開刀~

3、通用的ViewHolder

首先分析下ViewHolder的作用,通過convertView.setTag與convertView進(jìn)行綁定,然后當(dāng)convertView復(fù)用時(shí),直接從與之對(duì)于的ViewHolder(getTag)中拿到convertView布局中的控件,省去了findViewById的時(shí)間~

也就是說,實(shí)際上們每個(gè)convertView會(huì)綁定一個(gè)ViewHolder對(duì)象,這個(gè)viewHolder主要用于幫convertView存儲(chǔ)布局中的控件。

那么我們只要寫出一個(gè)通用的ViewHolder,然后對(duì)于任意的convertView,提供一個(gè)對(duì)象讓其setTag即可;

既然是通用,那么我們這個(gè)ViewHolder就不可能含有各種控件的成員變量了,因?yàn)槊總€(gè)Item的布局是不同的,最好的方式是什么呢?

提供一個(gè)容器,專門存每個(gè)Item布局中的所有控件,而且還要能夠查找出來;既然需要查找,那么ListView肯定是不行了,需要一個(gè)鍵值對(duì)進(jìn)行保存,鍵為控件的Id,值為控件的引用,相信大家立刻就能想到Map;但是我們不用Map,因?yàn)橛懈玫奶娲悾褪俏覀僡ndroid提供的SparseArray這個(gè)類,和Map類似,但是比Map效率,不過鍵只能為Integer.

下面看我們的ViewHolder類:

package com.example.zhy_baseadapterhelper; import android.content.Context; import android.util.Log; import android.util.SparseArray; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; public class ViewHolder { private final SparseArray<View> mViews; private View mConvertView; private ViewHolder(Context context, ViewGroup parent, int layoutId, int position) { this.mViews = new SparseArray<View>(); mConvertView = LayoutInflater.from(context).inflate(layoutId, parent, false); //setTag mConvertView.setTag(this); } /** * 拿到一個(gè)ViewHolder對(duì)象 * @param context * @param convertView * @param parent * @param layoutId * @param position * @return */ public static ViewHolder get(Context context, View convertView, ViewGroup parent, int layoutId, int position) { if (convertView == null) { return new ViewHolder(context, parent, layoutId, position); } return (ViewHolder) convertView.getTag(); } /** * 通過控件的Id獲取對(duì)于的控件,如果沒有則加入views * @param viewId * @return */ public <T extends View> T getView(int viewId) { View view = mViews.get(viewId); if (view == null) { view = mConvertView.findViewById(viewId); mViews.put(viewId, view); } return (T) view; } public View getConvertView() { return mConvertView; } }


與傳統(tǒng)的ViewHolder不同,我們使用了一個(gè)SparseArray<View>用于存儲(chǔ)與之對(duì)于的convertView的所有的控件,當(dāng)需要拿這些控件時(shí),通過getView(id)進(jìn)行獲取;

下面看使用該ViewHolder的MyAdapter;

@Override public View getView(int position, View convertView, ViewGroup parent) { //實(shí)例化一個(gè)viewHolder ViewHolder 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)); return viewHolder.getConvertView(); }

只看getView,其他方法都一樣;首先調(diào)用ViewHolder的get方法,如果convertView為null,new一個(gè)ViewHolder實(shí)例,通過使用mInflater.inflate加載布局,然后new一個(gè)SparseArray用于存儲(chǔ)View,最后setTag(this);

如果存在那么直接getTag

最后通過getView(id)獲取控件,如果存在則直接返回,否則調(diào)用findViewById,返回存儲(chǔ),返回。

好了,一個(gè)通用的ViewHolder寫好了,以后一個(gè)項(xiàng)目幾十個(gè)Adapter一個(gè)ViewHolder直接hold住全場~~大家可以省點(diǎn)時(shí)間斗個(gè)小地主了~~

4、打造通用的Adapter

有了通用的ViewHolder大家肯定不能滿足,怎么也得省出dota的時(shí)間,人在塔在~~

下面看如何打造一個(gè)通過的Adapter,我們叫做CommonAdapter

繼續(xù)分析,Adapter一般需要保持一個(gè)List對(duì)象,存儲(chǔ)一個(gè)Bean的集合,不同的ListView,Bean肯定是不同的,這個(gè)CommonAdapter肯定需要支持泛型,內(nèi)部維持一個(gè)List<T>,就解決我們的問題了;

于是我們初步打造我們的CommonAdapter

package com.example.zhy_baseadapterhelper; import java.util.List; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.TextView; public abstract class CommonAdapter<T> extends BaseAdapter { protected LayoutInflater mInflater; protected Context mContext; protected List<T> mDatas; public CommonAdapter(Context context, List<T> mDatas) { mInflater = LayoutInflater.from(context); this.mContext = context; this.mDatas = mDatas; } @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; } }
我們的CommonAdapter依然是一個(gè)抽象類,除了getView以外我們把其他的代碼都實(shí)現(xiàn)了,這樣的話,在使用我們的Adapter只要實(shí)現(xiàn)一個(gè)getView,然后getView里面再使用我們打造的通過的ViewHolder是不是感覺還不錯(cuò)~

現(xiàn)在我們的MyAdapter是這樣的:

package com.example.zhy_baseadapterhelper; import java.util.List; import android.content.Context; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; public class MyAdapter<T> extends CommonAdapter<T> { public MyAdapter(Context context, List<T> mDatas) { super(context, mDatas); } @Override public View getView(int position, 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)); return viewHolder.getConvertView(); } }

所有的代碼加起來也就10行左右,是不是神清氣爽~~稍等,我先去dota一把~

但是我們是否就這樣滿足了呢?顯然還可以簡化。

5、進(jìn)一步鑄造

注意我們的getView里面的代碼,雖然只有4行,但是我覺得所有的Adapter的

第一行(ViewHolder viewHolder = getViewHolder(position, convertView,parent);)和

最后一行:return viewHolder.getConvertView();一定是一樣的。

那么我們可以這樣做:我們把第一行和最后一行寫死,把中間變化的部分抽取出來,這不就是OO的設(shè)計(jì)原則嘛。現(xiàn)在CommonAdapter是這樣的:

package com.example.zhy_baseadapterhelper; import java.util.List; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; public abstract class n<T> extends BaseAdapter { protected LayoutInflater mInflater; protected Context mContext; protected List<T> mDatas; protected final int mItemLayoutId; public CommonAdapter(Context context, List<T> mDatas, int itemLayoutId) { this.mContext = context; this.mInflater = LayoutInflater.from(mContext); this.mDatas = mDatas; this.mItemLayoutId = itemLayoutId; } @Override public int getCount() { return mDatas.size(); } @Override public T 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) { final ViewHolder viewHolder = getViewHolder(position, convertView, parent); convert(viewHolder, getItem(position)); return viewHolder.getConvertView(); } public abstract void convert(ViewHolder helper, T item); private ViewHolder getViewHolder(int position, View convertView, ViewGroup parent) { return ViewHolder.get(mContext, convertView, parent, mItemLayoutId, position); } }

對(duì)外公布了一個(gè)convert方法,并且還把viewHolder和本Item對(duì)于的Bean對(duì)象給傳出去,現(xiàn)在convert方法里面需要干嘛呢?

通過ViewHolder把View找到,通過Item設(shè)置值;

現(xiàn)在我覺得代碼簡化到這樣,我已經(jīng)不需要單獨(dú)寫一個(gè)Adapter了,直接MainActivity匿名內(nèi)部類走起~

@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mListView = (ListView) findViewById(R.id.id_lv_main); //設(shè)置適配器 mListView.setAdapter(mAdapter = new CommonAdapter<String>( getApplicationContext(), mDatas, R.layout.item_single_str) { @Override public void convert(ViewHolder c, String item) { TextView view = viewHolder.getView(R.id.id_tv_title); view.setText(item); } }); }

可以看到效果咋樣,不錯(cuò)吧。你覺得還能簡化么?我覺得還能改善。

6、Adapter最后的封魔

我們現(xiàn)在在convertView里面需要這樣:

@Override
public void convert(ViewHolder viewHolder, String item)
{
TextView view = viewHolder.getView(R.id.id_tv_title);
view.setText(item);
}

我們細(xì)想一下,其實(shí)布局里面的View常用也就那么幾種:ImageView,TextView,Button,CheckBox等等;

那么我覺得ViewHolder還可以封裝一些常用的方法,比如setText(id,String);setImageResource(viewId, resId);setImageBitmap(viewId, bitmap);

那么現(xiàn)在ViewHolder是:

package com.example.zhy_baseadapterhelper; import android.content.Context; import android.graphics.Bitmap; import android.util.SparseArray; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.ImageView; import android.widget.TextView; import com.example.zhy_baseadapterhelper.ImageLoader.Type; public class ViewHolder { private final SparseArray<View> mViews; private int mPosition; private View mConvertView; private 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); // setTag mConvertView.setTag(this); } /** * 拿到一個(gè)ViewHolder對(duì)象 * * @param context * @param convertView * @param parent * @param layoutId * @param position * @return */ public static ViewHolder get(Context context, View convertView, ViewGroup parent, int layoutId, int position) { if (convertView == null) { return new ViewHolder(context, parent, layoutId, position); } return (ViewHolder) convertView.getTag(); } public View getConvertView() { return mConvertView; } /** * 通過控件的Id獲取對(duì)于的控件,如果沒有則加入views * * @param viewId * @return */ public <T extends View> T getView(int viewId) { View view = mViews.get(viewId); if (view == null) { view = mConvertView.findViewById(viewId); mViews.put(viewId, view); } return (T) view; } /** * 為TextView設(shè)置字符串 * * @param viewId * @param text * @return */ public ViewHolder setText(int viewId, String text) { TextView view = getView(viewId); view.setText(text); return this; } /** * 為ImageView設(shè)置圖片 * * @param viewId * @param drawableId * @return */ public ViewHolder setImageResource(int viewId, int drawableId) { ImageView view = getView(viewId); view.setImageResource(drawableId); return this; } /** * 為ImageView設(shè)置圖片 * * @param viewId * @param drawableId * @return */ public ViewHolder setImageBitmap(int viewId, Bitmap bm) { ImageView view = getView(viewId); view.setImageBitmap(bm); return this; } /** * 為ImageView設(shè)置圖片 * * @param viewId * @param drawableId * @return */ public ViewHolder setImageByUrl(int viewId, String url) { ImageLoader.getInstance(3, Type.LIFO).loadImage(url, (ImageView) getView(viewId)); return this; } public int getPosition() { return mPosition; } }

現(xiàn)在的MainActivity只需要這么寫:

mAdapter = new CommonAdapter<String>(getApplicationContext(), R.layout.item_single_str, mDatas) { @Override protected void convert(ViewHolder viewHolder, String item) { viewHolder.setText(R.id.id_tv_title, item); } };

convertView里面只要一行代碼了~~~

好了,到此我們的通用的Adapter已經(jīng)一步一步鑄造完畢~咋樣,以后寫項(xiàng)目省下來的時(shí)間是不是可以陪我切磋dota了(ps:11昵稱:血魔哥404)~~

注:關(guān)于ViewHolder里面的setText,setImageResource這類的方法,大家可以在使用的過程中不斷的完善,今天發(fā)現(xiàn)這個(gè)控件可以這么設(shè)置值,好,放進(jìn)去;時(shí)間長了,基本就完善了。還有那個(gè)ImageLoader是我另一篇博客里的,大家可以使用UIL,Volley或者自己寫個(gè)圖片加載器;

7、實(shí)踐

說了這么多,還是得拿出來讓我們的實(shí)踐檢驗(yàn)檢驗(yàn),順便來幾張?zhí)讏D,俗話說,沒圖沒正相。

1、我們的實(shí)例代碼的圖是這樣的:

關(guān)于Adapter和ViewHolder的代碼是這樣的:

// 設(shè)置適配器 mListView.setAdapter(mAdapter = new CommonAdapter<String>( getApplicationContext(), mDatas, R.layout.item_single_str) { @Override public void convert(ViewHolder helper, String item) { helper.setText(R.id.id_tv_title,item); } });

哎喲,我是不是只要貼一行;

2、來個(gè)復(fù)雜點(diǎn)的布局

<?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="wrap_content" android:background="#ffffff" android:orientation="vertical" android:padding="10dp" > <TextView android:id="@+id/tv_title" android:layout_width="match_parent" android:layout_height="wrap_content" android:singleLine="true" android:text="紅色錢包" android:textSize="16sp" android:textColor="#444444" > </TextView> <TextView android:id="@+id/tv_describe" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_below="@id/tv_title" android:layout_marginTop="10dp" android:maxLines="2" android:minLines="1" android:text="周三早上丟失了紅色錢包,在食堂二樓" android:textColor="#898989" android:textSize="16sp" > </TextView> <TextView android:id="@+id/tv_time" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_below="@id/tv_describe" android:layout_marginTop="10dp" android:text="20130240122" android:textColor="#898989" android:textSize="12sp" > </TextView> <TextView android:id="@+id/tv_phone" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_below="@id/tv_describe" android:layout_marginTop="10dp" android:background="#5cbe6c" android:drawableLeft="@drawable/icon_photo" android:drawablePadding="5dp" android:paddingBottom="3dp" android:paddingLeft="5dp" android:paddingRight="5dp" android:paddingTop="3dp" android:text="138024249542" android:textColor="#ffffff" android:textSize="12sp" > </TextView> </RelativeLayout>

效果圖是這樣的:


布局是不是挺復(fù)雜的了~~

但是代碼是這樣的:

// 設(shè)置適配器 mListView.setAdapter(mAdapter = new CommonAdapter<Bean>( getApplicationContext(), mDatas, R.layout.item_list) { @Override public void convert(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) } });

從一個(gè)字符串的布局到這樣的布局,Adapter加ViewHolder的改變就這么多,加起來3行左右代碼~~~


到此,Android 快速開發(fā)系列 打造萬能的ListView GridView 適配器結(jié)束;


最后給大家推薦一個(gè)gitHub項(xiàng)目:https://github.com/JoanZapata/base-adapter-helper ,這個(gè)項(xiàng)目所做的,和我上面寫的基本一致。

還有上面的布局文件來自網(wǎng)絡(luò),感謝Bmob的提供~

好了,我要去快樂的玩耍了~~


源碼點(diǎn)擊下載







生活不易,碼農(nóng)辛苦
如果您覺得本網(wǎng)站對(duì)您的學(xué)習(xí)有所幫助,可以手機(jī)掃描二維碼進(jìn)行捐贈(zèng)
程序員人生
------分隔線----------------------------
分享到:
------分隔線----------------------------
關(guān)閉
程序員人生
主站蜘蛛池模板: 最近的免费中文字幕视频 | 国产精品无码久久综合网 | 成人毛片一区二区三区 | 精品国产第一国产综合精品 | xxxx69欧美hdxxxhd| 毛片毛片| 国产欧美久久久另类精品 | 韩国理论片在线观看 | 久久久久久免费播放一级毛片 | 亚洲欧美乱综合图片区小说区 | 久夜色精品国产一区二区三区 | 亚洲最大中文字幕 | 欧美一级毛片不卡免费观看 | 国产一级做人爱c黑人版 | 国产欧美日韩精品一区二区三区 | 逼逼网| 一级毛片在线 | 亚洲精品欧美精品 | 欧美一区二区aa大片 | 亚洲图片综合网 | 亚洲五月七月丁香缴情 | 中文亚洲日韩欧美 | 中文字幕第23页 | 日韩国产欧美在线观看 | 亚洲大片在线观看 | 亚洲国产一区二区三区四区五区 | 亚洲欧洲日本精品 | 最新欧美精品 | 国产亚洲一区二区精品 | 一级做a级爰片性色毛片视频 | 精品中文字幕一区二区三区四区 | 国内精品久久久久影院嫩草 | 婷婷综合缴情亚洲五月伊 | 国产视频一二区 | 豆国产97在线 | 欧洲 | 最近最新中文字幕在线手机版 | 欧美一区二区三区在线播放 | 日韩精品欧美激情国产一区 | 欧美午夜在线播放 | 亚洲欧美日韩精品一区 | 亚洲 校园 春色 另类 激情 |