Android實(shí)戰(zhàn)RecyclerView頭部尾部添加方法示例
最近開(kāi)啟SDK Manager,突然發(fā)現(xiàn)android7.0的都有了,這迭代升級(jí)還真快。不過(guò)國(guó)內(nèi)普遍手機(jī)還是停留在4.4+,多則是是處于5.0版本的。Android5.0變化非常大,引入material design,加強(qiáng)權(quán)限管理、減少功耗...好像扯遠(yuǎn)了0 0?,F(xiàn)在直接進(jìn)入主題。在這里先感謝讀者的支持!!
ListView是有addHeaderView和 addFooterView兩個(gè)方法的.
但是作為官方推薦的ListView的升級(jí)版RecyclerView缺無(wú)法實(shí)現(xiàn)這兩個(gè)方法。
那么如果使用RecyclerView實(shí)現(xiàn)這兩個(gè)方法的效果該怎么做呢?
網(wǎng)上查詢了很久,試過(guò)各種各樣的實(shí)現(xiàn)方式,終于讓我發(fā)現(xiàn)一個(gè)還不錯(cuò)的實(shí)現(xiàn)方法,那么就給大家推薦一下。
筆者前陣子寫(xiě)了一個(gè)萬(wàn)能適配器,提供了上拉加載、上拉刷新的基礎(chǔ)功能,重要的是一個(gè)基礎(chǔ)baseAdapter能夠支持ListView與RecyclerView,后期提供傳送門(mén),現(xiàn)在我打算一步驟一步驟講下我的實(shí)現(xiàn)思路。
實(shí)戰(zhàn)RecyclerView頭部尾部添加方法
效果圖如下:


一、前提
首先ListView與RecyclerView兩者非常相似,兩者提供view都是依賴(lài)適配器。只不過(guò)就是5.0版本推出RecyclerView后,Google將adapter和viewHolder做了一系列的優(yōu)化和封裝。不像之前為了復(fù)用Listview里面的converView,要類(lèi)似在getView里面實(shí)現(xiàn)下列的代碼:

上面代碼看起來(lái)挺眼熟吧~
二、對(duì)比RecyclerView,google進(jìn)行的優(yōu)化
在RecyclerView依賴(lài)的適配器中,無(wú)論是適配器還是ViewHolder,從源碼我們可以看出,都存在RecyclerView的匿名內(nèi)部類(lèi)。相對(duì)于Listview,RecyclerView內(nèi)置了多級(jí)緩存、RecyclerViewPool(從線程的角度,可以理解成類(lèi)似線程池的東西,即多個(gè)RecyclerView可以公用一個(gè)view)、ViewHolder(已經(jīng)實(shí)現(xiàn)了復(fù)用,相對(duì)于Listview的BaseAdapter中g(shù)etView方法需要開(kāi)發(fā)者自己引入復(fù)用問(wèn)題方便很多)等等。這里我們簡(jiǎn)單說(shuō)下兩個(gè)方法:
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType)
public void onBindViewHolder(ViewHolder holder, int position)
在以前的BaseAdapter中,所有視圖加載、數(shù)據(jù)綁定以及復(fù)用,都需要我們直接在getView里面進(jìn)行操作。onCreateViewHolder負(fù)責(zé)視圖加載并且內(nèi)部完成復(fù)用,onBindViewHolder負(fù)責(zé)數(shù)據(jù)綁定并且內(nèi)部完成一系列的緩存機(jī)制。這里滿足了視圖層與邏輯層的分離,典型的mvp模式。
三、RecyclerView的頭部與尾部實(shí)現(xiàn)
RecyclerView不像ListView擁有addHeaderView()與addFooterView()的方法簡(jiǎn)單添加頭部尾部即可,而且RecyclerView也沒(méi)有像ListView的列表點(diǎn)擊監(jiān)聽(tīng)方法(setItemOnclickListener),這里我也不明白為什么官方會(huì)取消了這些獨(dú)有的屬性,不過(guò)我們依然可以在onBindViewHolder方法中進(jìn)行事件綁定!
具體頭部與尾部實(shí)現(xiàn)方法,這里有個(gè)訣竅,這里先看一個(gè)方法:
public int getItemViewType(int position)
getItemViewType方法是在執(zhí)行onCreateViewHolder(ViewGroup parent, int viewType)前回調(diào)用viewType,目的是為了根據(jù)viewType不同創(chuàng)建不同的視圖。我們可以通過(guò)在onCreateViewHolder創(chuàng)建視圖的時(shí)候,對(duì)viewType進(jìn)行判斷,如果添加了頭部,在position = 0的時(shí)候回調(diào)頭部的viewType給onCreateViewHolder,從而創(chuàng)建頭部。尾部創(chuàng)建方法于此類(lèi)同,直接看下代碼,適配器的實(shí)現(xiàn):
package cn.wsy.recyclerdemo;
import android.content.Context;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
/**
* Created by wsy on 2016/8/4.
*/
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyHolder> {
private RecyclerView mRecyclerView;
private List<String> data = new ArrayList<>();
private Context mContext;
private View VIEW_FOOTER;
private View VIEW_HEADER;
//Type
private int TYPE_NORMAL = 1000;
private int TYPE_HEADER = 1001;
private int TYPE_FOOTER = 1002;
public MyAdapter(List<String> data, Context mContext) {
this.data = data;
this.mContext = mContext;
}
@Override
public MyAdapter.MyHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if (viewType == TYPE_FOOTER) {
return new MyHolder(VIEW_FOOTER);
} else if (viewType == TYPE_HEADER) {
return new MyHolder(VIEW_HEADER);
} else {
return new MyHolder(getLayout(R.layout.item_list_layout));
}
}
@Override
public void onBindViewHolder(MyHolder holder, int position) {
if (!isHeaderView(position) && !isFooterView(position)) {
if (haveHeaderView()) position--;
TextView content = (TextView) holder.itemView.findViewById(R.id.item_content);
TextView time = (TextView) holder.itemView.findViewById(R.id.item_time);
content.setText(data.get(position));
time.setText("2016-1-1");
}
}
@Override
public int getItemCount() {
int count = (data == null ? 0 : data.size());
if (VIEW_FOOTER != null) {
count++;
}
if (VIEW_HEADER != null) {
count++;
}
return count;
}
@Override
public int getItemViewType(int position) {
if (isHeaderView(position)) {
return TYPE_HEADER;
} else if (isFooterView(position)) {
return TYPE_FOOTER;
} else {
return TYPE_NORMAL;
}
}
@Override
public void onAttachedToRecyclerView(RecyclerView recyclerView) {
try {
if (mRecyclerView == null && mRecyclerView != recyclerView) {
mRecyclerView = recyclerView;
}
ifGridLayoutManager();
} catch (Exception e) {
e.printStackTrace();
}
}
private View getLayout(int layoutId) {
return LayoutInflater.from(mContext).inflate(layoutId, null);
}
public void addHeaderView(View headerView) {
if (haveHeaderView()) {
throw new IllegalStateException("hearview has already exists!");
} else {
//避免出現(xiàn)寬度自適應(yīng)
ViewGroup.LayoutParams params = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
headerView.setLayoutParams(params);
VIEW_HEADER = headerView;
ifGridLayoutManager();
notifyItemInserted(0);
}
}
public void addFooterView(View footerView) {
if (haveFooterView()) {
throw new IllegalStateException("footerView has already exists!");
} else {
ViewGroup.LayoutParams params = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
footerView.setLayoutParams(params);
VIEW_FOOTER = footerView;
ifGridLayoutManager();
notifyItemInserted(getItemCount() - 1);
}
}
private void ifGridLayoutManager() {
if (mRecyclerView == null) return;
final RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager();
if (layoutManager instanceof GridLayoutManager) {
final GridLayoutManager.SpanSizeLookup originalSpanSizeLookup =
((GridLayoutManager) layoutManager).getSpanSizeLookup();
((GridLayoutManager) layoutManager).setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
@Override
public int getSpanSize(int position) {
return (isHeaderView(position) || isFooterView(position)) ?
((GridLayoutManager) layoutManager).getSpanCount() :
1;
}
});
}
}
private boolean haveHeaderView() {
return VIEW_HEADER != null;
}
public boolean haveFooterView() {
return VIEW_FOOTER != null;
}
private boolean isHeaderView(int position) {
return haveHeaderView() && position == 0;
}
private boolean isFooterView(int position) {
return haveFooterView() && position == getItemCount() - 1;
}
public static class MyHolder extends RecyclerView.ViewHolder {
public MyHolder(View itemView) {
super(itemView);
}
}
}
四、實(shí)現(xiàn)方法
簡(jiǎn)單的初始化RecycerView,以及設(shè)置適配器,如下:
private void initRecyc() {
// mRecyclerView.setLayoutManager(new GridLayoutManager(this,2));
mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
adapter = new MyAdapter(data, this);
mRecyclerView.setAdapter(adapter);
adapter.addFooterView(LayoutInflater.from(this).inflate(R.layout.item_footer_layout,null));
adapter.addHeaderView(LayoutInflater.from(this).inflate(R.layout.item_header_layout,null));
}
五、注意的問(wèn)題
筆者在添加頭部尾部的時(shí)候,發(fā)現(xiàn)在配置RecyclerView,如果模式是配置GridLayoutManager的時(shí)候,發(fā)現(xiàn)頭部會(huì)跑到第一格,也就是不是自己想要獨(dú)立一行的效果,這里貼上關(guān)鍵代碼,可以解決(簡(jiǎn)單數(shù)學(xué)問(wèn)題啦哈~):
private void ifGridLayoutManager() {
if (mRecyclerView == null) return;
final RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager();
if (layoutManager instanceof GridLayoutManager) {
final GridLayoutManager.SpanSizeLookup originalSpanSizeLookup =
((GridLayoutManager) layoutManager).getSpanSizeLookup();
((GridLayoutManager) layoutManager).setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
@Override
public int getSpanSize(int position) {
return (isHeaderView(position) || isFooterView(position)) ?
((GridLayoutManager) layoutManager).getSpanCount() :
1;
}
});
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android實(shí)現(xiàn)程序自動(dòng)升級(jí)到安裝示例分享(下載android程序安裝包)
這篇文章主要介紹了android實(shí)現(xiàn)下載android程序安裝包自動(dòng)升級(jí)的示例,大家參考使用吧2014-01-01
Android自定義View實(shí)現(xiàn)shape圖形繪制
這篇文章主要為大家詳細(xì)介紹了Android使用自定義View實(shí)現(xiàn)shape圖形繪制,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01
Android應(yīng)用開(kāi)發(fā)之代碼混淆
Android項(xiàng)目中的混淆很easy,之所以寫(xiě)這篇總結(jié)是由于近期發(fā)現(xiàn)公司的代碼居然沒(méi)有混淆,反編譯后代碼隨手可得。很震驚。2014-07-07
Android實(shí)現(xiàn)卡片翻轉(zhuǎn)動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)卡片翻轉(zhuǎn)動(dòng)畫(huà),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01
Android WebView 常見(jiàn)問(wèn)題及處理方案
這篇文章主要介紹了Android WebView 常見(jiàn)問(wèn)題及處理方案,需要的朋友可以參考下2015-08-08
Android 使用ViewPager實(shí)現(xiàn)圖片左右循環(huán)滑動(dòng)自動(dòng)播放
這篇文章主要介紹了Android 使用ViewPager實(shí)現(xiàn)圖片左右循環(huán)滑動(dòng)自動(dòng)播放的相關(guān)資料,非常不錯(cuò),具有參考解決價(jià)值,需要的朋友可以參考下2016-08-08
淺談android性能優(yōu)化之啟動(dòng)過(guò)程(冷啟動(dòng)和熱啟動(dòng))
本篇文章主要介紹了淺談android性能優(yōu)化之啟動(dòng)過(guò)程(冷啟動(dòng)和熱啟動(dòng)) ,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-08-08

