學(xué)習(xí)Android開(kāi)發(fā)之RecyclerView使用初探
在進(jìn)行一些MaterialDesign規(guī)范開(kāi)發(fā)的時(shí)候,比如之前說(shuō)到的CoordinateLayout實(shí)現(xiàn)的向上折疊效果的時(shí)候,如果依然使用ListView,那么這種效果是做不出來(lái)的,因?yàn)長(zhǎng)istView不兼容這個(gè)控件,而替代它的就是RecyclerView。
和ListView的區(qū)別:
①RecyclerView只關(guān)心item的重用和緩存
②RecyclerView不關(guān)心item的分隔風(fēng)格(交給ItemDecoration)
③RecyclerView不關(guān)心item的動(dòng)畫(huà)(交給ItemAnimator)
④RecyclerView不關(guān)心item的布局(交給LayoutManager)
本文為初探,所以先不詳解②③④這幾個(gè)內(nèi)容,先記錄一下用法和遇到的坑,后續(xù)有時(shí)間再更新!
先放一下效果圖,動(dòng)畫(huà)效果在之前的博客也有提到,可以參考,主要的是中間的RecyclerView,這里主要講如何使用它以及遇到的一些問(wèn)題
先添加依賴,版本無(wú)所謂:
compile 'com.android.support:recyclerview-v7:23.1.1'
布局文件如下,注意最后一個(gè)屬性可以不用理,這是用來(lái)協(xié)調(diào)動(dòng)畫(huà)效果的(就是把上面圖片推上去的效果)
<android.support.v7.widget.RecyclerView android:id="@+id/main_lv" android:layout_width="match_parent" android:layout_height="match_parent" app:layout_behavior="@string/appbar_scrolling_view_behavior"/>
自定義一個(gè)Adapter繼承自RecyclerView.Adapter,代碼如下:
public class MyAdapter extends RecyclerView.Adapter { private static final String TAG = "MyAdapter"; private final List<Map<String, Object>> data; private final View root; private OnItemButtonCLickListener listener; public MyAdapter(List<Map<String, Object>> data, View root) { this.data = data; this.root = root; } public void setOnItemButtonClickListener(OnItemButtonCLickListener listener){ this.listener = listener; } class ViewHolder extends RecyclerView.ViewHolder { TextView reason, money, date; Button delete,alter; public ViewHolder(View itemView) { super(itemView); reason = (TextView) itemView.findViewById(R.id.main_rv_item_reason); money = (TextView) itemView.findViewById(R.id.main_rv_item_money); date = (TextView) itemView.findViewById(R.id.main_rv_item_date); delete = (Button) itemView.findViewById(R.id.main_rv_item_remove); alter = (Button) itemView.findViewById(R.id.main_rv_item_alter); } } @Override public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { ViewHolder vh = new ViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout .main_lv_item, null)); return vh; } @Override public void onBindViewHolder(RecyclerView.ViewHolder holder, final int position) { final ViewHolder vh = (ViewHolder) holder; vh.date.setText(data.get(position).get("date").toString()); vh.reason.setText(data.get(position).get("reason").toString()); vh.money.setText(data.get(position).get("money").toString()); vh.delete.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Snackbar snackBar = Snackbar.make(root, "Are you sure to delete it?", Snackbar.LENGTH_LONG); snackBar.setAction("YES", new MyOnClickListener(vh.getLayoutPosition())); snackBar.show(); } }); vh.alter.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { listener.OnItemAlter(vh.getLayoutPosition()); } }); } @Override public int getItemCount() { return data.size(); } private class MyOnClickListener implements View.OnClickListener { private final int position; public MyOnClickListener(int position) { this.position = position; } @Override public void onClick(View v) { listener.OnItemDelete(position); } } interface OnItemButtonCLickListener { void OnItemAlter(int position); void OnItemDelete(int position); } }
解析:構(gòu)造方法中獲取到數(shù)據(jù)源以及根View,根View用來(lái)顯示SnackBar,如果不知道SnackBar,請(qǐng)看上一篇。然后創(chuàng)建一個(gè)內(nèi)部類,繼承自RecyclerView.ViewHolder,用它來(lái)保存每一個(gè)Item的全部控件。接下來(lái)重寫(xiě)幾個(gè)方法:getItemCount、onBindViewHolder以及onCreateViewHolder,他們分別是返回item個(gè)數(shù),綁定ViewHolder以及創(chuàng)建ViewHolder,創(chuàng)建ViewHolder就是通過(guò)Item的布局文件獲取每個(gè)控件的實(shí)例,綁定ViewHolder就是把數(shù)據(jù)源中的數(shù)據(jù)映射到Item的控件中,因?yàn)槲覀兊腎tem中還有刪除和修改按鈕,所以我們?yōu)樗鼈兲砑狱c(diǎn)擊事件也是在onBindViewHolder中進(jìn)行。接下來(lái)我們定義一個(gè)接口OnItemButtonClickListener并且定義兩個(gè)方法分別用來(lái)回調(diào)點(diǎn)擊刪除和修改,并創(chuàng)建接口的對(duì)象,在按鈕點(diǎn)擊的時(shí)候分別調(diào)用接口的兩個(gè)方法,并傳入點(diǎn)擊的item的position。
Activity部分代碼如下:
rv = (RecyclerView) findViewById(R.id.main_lv); adapter = new MyAdapter(data, root); rv.setLayoutManager(new LinearLayoutManager(this,LinearLayoutManager.VERTICAL,false)); // 設(shè)置布局,否則無(wú)法正常使用 adapter.setOnItemButtonClickListener(this); rv.setAdapter(adapter);
@Override public void OnItemAlter(int position) { //Intent intent = new Intent(MainActivity.this, AlterActivity.class); //intent.putExtra("id", Integer.parseInt(data.get(position).get("id").toString())); //intent.putExtra("date", data.get(position).get("date").toString()); //intent.putExtra("money", data.get(position).get("money").toString()); //intent.putExtra("reason", data.get(position).get("reason").toString()); //startActivityForResult(intent, REQUEST_ALTER); } @Override public void OnItemDelete(int position) { Log.d(TAG, "OnItemDelete() called with: " + "position = [" + position + "]"); //Map<String, Object> backup = data.get(position); //data.remove(position); //database.execSQL("delete from tally where id=?", new String[]{backup.get // ("id").toString()}); adapter.notifyItemRemoved(position); }
解析:獲取RecyclerView的實(shí)例之后,設(shè)置對(duì)應(yīng)的布局管理器,然后通過(guò)setOnItemButtonClickListener設(shè)置回調(diào)函數(shù)并實(shí)現(xiàn),最后給RecyclerView設(shè)置adapter即可。
遇到的問(wèn)題:
①剛開(kāi)始忘記設(shè)置布局,RecyclerView不顯示出來(lái)
②因?yàn)槲覀冊(cè)趧h除item之后調(diào)用了notifyItemRemoved方法來(lái)顯示動(dòng)畫(huà)并刷新,在onBindViewHolder中,對(duì)刪除按鈕監(jiān)聽(tīng)之后調(diào)用接口方法的時(shí)候傳入了position,這個(gè)position是錯(cuò)誤的,因?yàn)樵诮壎ǖ臅r(shí)候固定了,如果數(shù)據(jù)源的數(shù)據(jù)被刪除了,數(shù)據(jù)源的size會(huì)變小,這個(gè)position就不再對(duì)應(yīng)于數(shù)據(jù)源的每一個(gè)項(xiàng)就會(huì)產(chǎn)生混亂,所以這里應(yīng)該傳入viewholder.getLayoutPosition()所得到的值,這個(gè)值是對(duì)應(yīng)點(diǎn)擊而產(chǎn)生的。
③如果在RecyclerView的第一項(xiàng)前面插入一個(gè)item,并且調(diào)用notifyItmeInserted(0),是不會(huì)播放動(dòng)畫(huà)效果的。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android RecyclerView 基礎(chǔ)知識(shí)詳解
- Android RecyclerView加載不同布局簡(jiǎn)單實(shí)現(xiàn)
- Android添加圖片到ListView或者RecyclerView顯示
- Android中RecyclerView實(shí)現(xiàn)橫向滑動(dòng)代碼
- Android RecyclerView詳解之實(shí)現(xiàn) ListView GridView瀑布流效果
- Android中RecyclerView點(diǎn)擊Item設(shè)置事件
- Android RecyclerView藝術(shù)般的控件使用完全解析
- Android代碼實(shí)現(xiàn)AdapterViews和RecyclerView無(wú)限滾動(dòng)
- Android RecyclerView滑動(dòng)刪除和拖動(dòng)排序
- Android RecyclerView的Item自定義動(dòng)畫(huà)及DefaultItemAnimator源碼分析
- Android開(kāi)發(fā)中RecyclerView組件使用的一些進(jìn)階技講解
- Android RecyclerView 數(shù)據(jù)綁定實(shí)例代碼
相關(guān)文章
Android實(shí)現(xiàn)頂部導(dǎo)航菜單左右滑動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)頂部導(dǎo)航菜單左右滑動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06Android Handler runWithScissors 梳理流程解析
這篇文章主要為大家介紹了Android Handler runWithScissors 梳理流程解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10Android判斷服務(wù)是否運(yùn)行及定位問(wèn)題實(shí)例分析
這篇文章主要介紹了Android判斷服務(wù)是否運(yùn)行及定位問(wèn)題,以實(shí)例形式較為詳細(xì)的分析了Android判斷服務(wù)運(yùn)行狀態(tài)及獲取經(jīng)緯度的相關(guān)實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-09-09Android基于Sqlite實(shí)現(xiàn)注冊(cè)和登錄功能
這篇文章主要為大家詳細(xì)介紹了Android基于Sqlite實(shí)現(xiàn)注冊(cè)和登錄功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Android實(shí)現(xiàn)儀表盤(pán)控件開(kāi)發(fā)
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)儀表盤(pán)控件開(kāi)發(fā),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05