Android中對(duì)RecyclerView Adapter封裝解析
前言
關(guān)于adapter的封裝,網(wǎng)上有很多開(kāi)源庫(kù),開(kāi)發(fā)的時(shí)候可以直接拿來(lái)用,省了很多事。
最近閑來(lái)無(wú)事,想著自己動(dòng)手封裝一個(gè)adapter。
問(wèn)題
1、通常我們封裝的時(shí)候,可以簡(jiǎn)化到這一步:
BaseRecyclerViewAdapter adapter = new BaseRecyclerViewAdapter() {
private static final int TYPE_FIR = 1;
private static final int TYPE_SEC = 2;
private static final int TYPE_THR = 3;
@Override
public int getLayoutId(int viewType) {
if (viewType == TYPE_FIR) {
return R.layout.test_activity_recyclerview_item;
} else if (viewType == TYPE_SEC){
return R.layout.test_activity_recyclerview_item_two;
} else {
return R.layout.test_activity_recyclerview_item_three;
}
}
@Override
public int getItemViewType(int position) {
if (position % 3 == 1) {
return TYPE_FIR;
} else if (position % 3 == 2) {
return TYPE_SEC;
} else {
return TYPE_THR;
}
}
@Override
public void onBindRecyclerViewHolder(BaseViewHolder holder, int position) {
if (getItemVIewType(position) == TYPE_FIR) {
// TODO 數(shù)據(jù)處理及綁定
} else if (getItemVIewType(position) == TYPE_SEC) {
// TODO 數(shù)據(jù)處理及綁定
} else {
// TODO 數(shù)據(jù)處理及綁定
}
}
};
2、從上面代碼我們可以看到,當(dāng)處理多布局類型的時(shí)候,我們需要解決的是:
- 多布局類型定義
- 返回什么類型處理
- 根據(jù)指定類型,加載對(duì)應(yīng)布局Layout
- 根據(jù)類型,處理及綁定數(shù)據(jù)
3、當(dāng)我們迭代更新的時(shí)候,我們只能通過(guò)修改adapter,這樣耦合性稍微有點(diǎn)強(qiáng)。
不過(guò)呢,簡(jiǎn)化到這一步的時(shí)候,其實(shí)在項(xiàng)目中是可以使用的
解決
1、上面的問(wèn)題,我們知道,如果需要迭代更新,我們只能通過(guò)修改viewType、layout、onBind等具體數(shù)據(jù)來(lái)實(shí)現(xiàn)。既然這樣,我們可以將這些具體數(shù)據(jù)抽取封裝起來(lái),adapter不直接處理數(shù)據(jù),而是交給這個(gè)封裝類,邏輯交給封裝類來(lái)處理,這樣adapter只需要負(fù)責(zé)維護(hù)這個(gè)封裝類即可。
2、這個(gè)封裝類,我們暫且稱作Item
interface Item {
@LayoutRes
int getLayoutResource();
int getItemViewType();
BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType);
void onBindViewHolder(BaseViewHolder holder, int position);
}
adpater中,需要?jiǎng)討B(tài)調(diào)整的就是
- viewType
- layout布局
- onCreateViewHolder
- onBindViewHolder
所以就將這些方法抽取封裝起來(lái),然后adapter維護(hù)Item列表即可
3、上面這個(gè)接口Item
第一:沒(méi)有數(shù)據(jù),我們需要自己在實(shí)現(xiàn)類中提供數(shù)據(jù)設(shè)置獲取方法;
第二:我們需要同時(shí)實(shí)現(xiàn)四個(gè)方法
既然這樣,我們可以通過(guò)一個(gè)基類BaseItem,來(lái)簡(jiǎn)化操作
4、在Item接口中,只有onCreateViewHolder()方法,是不需要每次都實(shí)現(xiàn)的,因?yàn)樗枪潭ú蛔兊牡?,BaseItem就可以這樣寫(xiě):
public abstract class BaseItem<T> implements Item {
public T mData;
public void setData(T t) {
this.mData = t;
}
@Override
public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
int resourceId = getLayoutResource();
View itemView = LayoutInflater.from(parent.getContext()).inflate(resourceId, parent, false);
return new BaseViewHolder(itemView);
}
}
5、現(xiàn)在,我們只需要實(shí)現(xiàn)三個(gè)方法就可以了。我們?cè)倏磄etItemViewType,這個(gè)返回的是當(dāng)前布局類型,int類型,唯一且不能重復(fù) 。Android中的Layout維護(hù)著一個(gè)int型的唯一標(biāo)識(shí)ID,我們是不是可以拿這個(gè)標(biāo)識(shí)ID,作為viewType?這樣就可以進(jìn)一步簡(jiǎn)化實(shí)現(xiàn)類,現(xiàn)在我們只需實(shí)現(xiàn)兩個(gè)方法即可。
public abstract class BaseItem<T> implements Item {
public T mData;
public void setData(T t) {
this.mData = t;
}
@Override
public int getItemViewType() {
return getLayoutResource();
}
@Override
public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
int resourceId = getLayoutResource();
View itemView = LayoutInflater.from(parent.getContext()).inflate(resourceId, parent, false);
return new BaseViewHolder(itemView);
}
}
6、接下來(lái),我們要構(gòu)造Adapter,使之不再維護(hù)具體數(shù)據(jù),而是BaseItem列表
public class BaseAdapter extends RecyclerView.Adapter<BaseViewHolder> {
private List<BaseItem> mData;
public BaseRecyclerViewAdapter(List<BaseItem> data) {
mData = new ArrayList<>();
if (data != null) {
mData.addAll(data);
}
}
@Override
public BaseViewHolder onCreateViewHolder(ViewGroup parent, final int viewType) {
for (BaseItem item : mData) {
if (viewType == item.getItemViewType()) {
return item.onCreateViewHolder(parent, viewType);;
}
}
}
@Override
public void onBindViewHolder(BaseViewHolder holder, int position) {
mData.get(dataPosition).onBindViewHolder(holder, position);
}
@Override
public int getItemCount() {
return mData.size();
}
@Override
public int getItemViewType(int position) {
return mData.get(position).getItemViewType();
}
}
結(jié)語(yǔ)
以上簡(jiǎn)單介紹了一下封裝思路,至于像頭布局、腳布局、空布局、加載更多布局等等,其實(shí)都是Adapter中的一種vietType布局,具體實(shí)現(xiàn)代碼,GitHub:BaseAdapter
下圖是一adapter中數(shù)據(jù)構(gòu)造:

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Flow轉(zhuǎn)LiveData數(shù)據(jù)丟失原理詳解
這篇文章主要為大家介紹了Flow轉(zhuǎn)LiveData數(shù)據(jù)丟失原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
如何判斷軟件程序是否聯(lián)網(wǎng) 聯(lián)網(wǎng)狀態(tài)提示信息Android實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了如何判斷軟件程序是否聯(lián)網(wǎng)的實(shí)現(xiàn)代碼,Android實(shí)現(xiàn)聯(lián)網(wǎng)狀態(tài)信息提示,感興趣的小伙伴們可以參考一下2016-05-05
當(dāng)ListView有Header時(shí) onItemClick里的position不正確的原因
這篇文章主要介紹了當(dāng)ListView有Header時(shí) onItemClick里的position不正確的原因的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07
解析Android 8.1平臺(tái)SystemUI 導(dǎo)航欄加載流程
這篇文章主要介紹了Android 8.1平臺(tái)SystemUI 導(dǎo)航欄加載流程,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-11-11
Android實(shí)現(xiàn)ListView異步加載的方法(改進(jìn)版)
這篇文章主要介紹了Android實(shí)現(xiàn)ListView異步加載的方法,針對(duì)前面介紹的方法進(jìn)行了線程操作的改進(jìn),具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08
Android開(kāi)發(fā)筆記之: 數(shù)據(jù)存儲(chǔ)方式詳解
本篇文章是對(duì)Android中數(shù)據(jù)存儲(chǔ)方式進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
Android開(kāi)發(fā)-之監(jiān)聽(tīng)button點(diǎn)擊事件的多種方法
本篇文章主要是介紹了Android開(kāi)發(fā)之監(jiān)聽(tīng)button點(diǎn)擊事件的方法,Android開(kāi)發(fā)-之監(jiān)聽(tīng)button點(diǎn)擊事件的方法總結(jié),有興趣的可以了解一下。2016-11-11

