ListView通用泛型適配器
還記得我們之前說的ListView嗎,(這個難用的控件-。+)我們在用他的同時也用到了一個叫做適配器Adapter的東西。一般我們用一個類繼承BaseAdapter,來進(jìn)行數(shù)據(jù)和控件的適配。
但是我們每一種適配器都只是為了適配一種數(shù)據(jù)源和一種布局,如果用到的少還好,如果要用到十幾種,我們是不是要寫十幾個適配器呢?這個想法真的是太蠢了!
有一種適配器寫法,可以做到一個適配器與多種類型數(shù)據(jù)和布局進(jìn)行適配,這個東西叫做通用適配器(因為他是用到泛型實現(xiàn)的,我稱他為泛型適配器),今天我們來看一下這種適配器的寫法:
在寫之前呢,我們首先回憶一下之前所用到的BaseAdapter適配器:
我們通過繼承BaseAdapter,實現(xiàn)了他的四個方法:getCount,getPosition,getItem,和getView。其中最難寫的就是getView了,然后我們還對他進(jìn)行了優(yōu)化:通過寫一個叫做ViewHolder的類,在里面放入對應(yīng)的控件。
現(xiàn)在我們首先來說一下通用適配器和一般的適配器的區(qū)別和相同點:
接下來我們正式來看一下通用適配器的寫法:
1.先創(chuàng)建好我們今天需要的控件、源數(shù)據(jù)以及Bean類。
控件只有一個ListView
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".activities.MainActivity"> <ListView android:layout_width="match_parent" android:layout_height="match_parent" android:id="@+id/mlv"/> </LinearLayout>
beans方法模擬了一個假數(shù)據(jù)
private void beans() { list = new ArrayList<>(); for (int i = 0; i < 16; i += 4) { list.add(new Student("同學(xué)" + i, "男", 15 + i, R.drawable.a, true)); list.add(new Student("同學(xué)" + (i + 1), "男", 15 + i, R.drawable.b, false)); list.add(new Student("同學(xué)" + (i + 2), "男", 15 + i, R.drawable.c, false)); list.add(new Student("同學(xué)" + (i + 3), "男", 15 + i, R.drawable.d, true)); } }
這是bean類
package zy.pers.homework_20.bean; public class Student { private String name; private String sex; private int age; private int imgId; private boolean isOver; public Student(String name, String sex, int age, int imgId,boolean isOver) { this.name = name; this.sex = sex; this.age = age; this.imgId = imgId; this.isOver = isOver; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public int getImgId() { return imgId; } public void setImgId(int imgId) { this.imgId = imgId; } @Override public String toString() { return "Student{" + "name='" + name + '\'' + ", sex='" + sex + '\'' + ", age=" + age + ", imgId=" + imgId + '}'; } public boolean isOver() { return isOver; } public void setOver(boolean over) { isOver = over; } }
2.創(chuàng)建MyBaseAdapter繼承BaseAdapter
public class MyBaseAdapter<T> extends BaseAdapter { @Override public int getCount() { return 0; } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return 0; } @Override public View getView(int position, View convertView, ViewGroup parent) { return null; } }
3.我們說通用適配器傳入布局id和源數(shù)據(jù),所以我們定義這兩個量接收傳入的數(shù)據(jù)。
private List<Student> list; private int mLayRes; public MyBaseAdapter(List<Student> list, int mLayRes) { this.list= list; this.mLayRes = mLayRes; }
4.重寫我們的前三個方法
前三個方法應(yīng)該算是比較簡單的了,
@Override public int getCount() { return list != null ? list.size() : 0; } @Override public T getItem(int position) { return list.get(position); } @Override public long getItemId(int position) { return position; }
第一個一個簡單的判斷,返回list的大小。第二個跟第三個和以前適配器一樣,只是getItem的返回值寫成了泛型。
5.寫Viewholder類,這個是很麻煩的,我們先創(chuàng)建出來Viewholder,之后的方法我們一步一步添加。
public static class ViewHolder { private SparseArray<View> mViews = new SparseArray<>(); private Context mContext; private int position; private int layRes; private View itemView;
private ViewHolder(Context context, ViewGroup parent, int layRes) { this.mContext = context; this.layRes= layRes; this.itemView = LayoutInflater.from(context).inflate(layRes, parent, false); this.itemView.setTag(this); }
public static ViewHolder bind(int position, View convertView, ViewGroup parent, int layRes, Context context) { ViewHolder holder; if (convertView == null) { holder = new ViewHolder(context, parent, layRes); } else { holder = (ViewHolder) convertView.getTag(); holder.itemView = convertView; } holder.position = position; return holder; }
東西有點多,我們順著邏輯慢慢看:
1)首先是通過單例來實現(xiàn),所以我們需要一個私有化構(gòu)造方法,里面有三個參數(shù),分別是上下文,ViewGroup和布局id,這三個屬性是我們必須要用到的,我們傳入上下文獲取inflater,把布局id傳進(jìn)去,然后把holder傳入我們的itemView中。
這一步我們應(yīng)該比較熟悉吧,我們以前是在getView中實現(xiàn)這一步的。
2)然后我們看下面的bind方法,他的參數(shù)有五個。其實有三個參數(shù)我們很熟悉,就是我們getView中的三個參數(shù)。在這基礎(chǔ)上我們又添加了兩個參數(shù),布局id和上下文。
然后為了優(yōu)化我們先判斷當(dāng)前的convertView是否為空,如果為空就新建一個Viewholder,讓convertView在私有構(gòu)造器中加載;如果不為空,直接通過getTag拿到。
注意我們要對holder中的兩個參數(shù)進(jìn)行修改,一個是itemView,一個是position。因為我們優(yōu)化過后,如果convertView不為空,他里面是有之前的數(shù)據(jù)的,其他的幾個屬性我們不用管,但是這兩個還是儲存著上一個的內(nèi)容。我們需要讓他重新指向當(dāng)前的convertView和position,給大家畫一張圖就很明白了:
索引什么的畫的可能不準(zhǔn)確,但是主要就是這么個意思,大家領(lǐng)會精神哈。
最后返回holder。
3)我們還需要返回我們加載完成的convertView,
public View getItemView() { return itemView; }
現(xiàn)在我們Viewholder基本框架寫完了,我們暫時不管他了,去寫getView。
6.重寫方法getView:
我們剛才說了,在adapter中寫一個抽象方法,然后通過回調(diào)方法,實現(xiàn)多類型適配,也就是說這個抽象方法是寫我們給具體控件添加數(shù)據(jù)的,我們在這里面?zhèn)鬟f兩個參數(shù),一個是我們的Viewholder,另一個是對應(yīng)位置的數(shù)據(jù),類型為泛型。
public abstract void bindView(ViewHolder holder,T obj);
因為我們出現(xiàn)了抽象方法,所以我們的MyBaseAdapter需要變成抽象類,
public abstract class MyBaseAdapter<T> extends BaseAdapter {
這是我們的getView
@Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder = ViewHolder.bind(position,convertView,parent,mLayRes,parent.getContext()); bindView(holder,list.get(position)); return holder.getItemView(); }
現(xiàn)在我們的適配器已經(jīng)完成百分之九十了,還差一點,我們需要寫幾個輔助方法,為了方便我們等會進(jìn)行適配。
1.獲取指定控件
public <T extends View> T getView(int id){ T t = (T) mViews.get(id); if(t == null){ t = itemView.findViewById(id); mViews.put(id,t); } return t; }
在Viewholder中寫一個getView方法,通過控件id來獲取指定控件。
2.TextView控件輸入數(shù)據(jù)
public ViewHolder setText(int id,CharSequence text){ View view = getView(id); if(view instanceof View){ ((TextView)view).setText(text); } return this; }
3.ImageView輸入圖片
public ViewHolder setImg(int id,int resId){ View view = getView(id); if(view instanceof View){ ((ImageView)view).setImageResource(resId); }else view.setBackgroundResource(resId); return this; }
4.復(fù)選框輸入選定狀態(tài)
public ViewHolder setCheckable(int id,boolean checkable){ View view = getView(id); if(view instanceof View){ ((CheckBox)view).setChecked(checkable); } return this; }
好啦,先在我們的適配器完全寫完了,我們來看一下效果吧。
private void initTools() { ListView mLv = (ListView) findViewById(R.id.mlv); adapter = new MyBaseAdapter<Student>(list,R.layout.item_one) { @Override public void bindView(ViewHolder holder, Student obj) { holder.setText(R.id.name,obj.getName()) .setText(R.id.age,obj.getAge() + "") .setText(R.id.sex,obj.getSex()) .setImg(R.id.head,obj.getImgId()) .setCheckable(R.id.mc,obj.isOver()); } }; mLv.setAdapter(adapter); }
雖然效果有點丑,但是功能我們實現(xiàn)了哈哈,大家如果不信可以在創(chuàng)建一個新的bean類和新的layout布局試驗一下,同樣可以。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
- Android ListView和Adapter數(shù)據(jù)適配器的簡單介紹
- 詳解xamarin Android 實現(xiàn)ListView萬能適配器
- Android ListView適配器(Adapter)優(yōu)化方法詳解
- Android ListView萬能適配器實例代碼
- Android巧用XListView實現(xiàn)萬能下拉刷新控件
- android使用flutter的ListView實現(xiàn)滾動列表的示例代碼
- Android自定義控件ListView下拉刷新的代碼
- MVPXlistView展示上拉下拉效果
- Android通過代碼控制ListView上下滾動的方法
- Android解決ScrollView下嵌套ListView和GridView中內(nèi)容顯示不全的問題
相關(guān)文章
Android PhoneWindowManager監(jiān)聽屏幕右側(cè)向左滑動實現(xiàn)返回功能
這篇文章主要介紹了Android PhoneWindowManager監(jiān)聽屏幕右側(cè)向左滑動實現(xiàn)返回功能,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-04-04Android APT 實現(xiàn)控件注入框架SqInject的示例
這篇文章主要介紹了Android APT 實現(xiàn)控件注入框架SqInject的示例,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-03-03Android中TextView實現(xiàn)分段顯示不同顏色的字符串
在做項目的時候,遇到過一行文字有兩種顏色。在菜鳥的時候直接會想到用多個TextView來實現(xiàn),所以下面這篇文章主要給大家介紹了關(guān)于Android中TextView如何實現(xiàn)分段顯示不同顏色字符串的相關(guān)資料,需要的朋友可以參考下。2017-12-12Android用PopupWindow實現(xiàn)自定義Dailog
這篇文章主要為大家詳細(xì)介紹了Android用PopupWindow實現(xiàn)自定義Dailog的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-01-01Android中ACTION_CANCEL的觸發(fā)機制與滑出子view的情況
這篇文章主要介紹了Android中ACTION_CANCEL的觸發(fā)機制與滑出子view的情況,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09