學習Android開發之RecyclerView使用初探_Android

來源:互聯網
上載者:User

在進行一些MaterialDesign規範開發的時候,比如之前說到的CoordinateLayout實現的向上摺疊效果的時候,如果依然使用ListView,那麼這種效果是做不出來的,因為ListView不相容這個控制項,而替代它的就是RecyclerView。
和ListView的區別: 

①RecyclerView只關心item的重用和緩衝 
②RecyclerView不關心item的分隔風格(交給ItemDecoration) 
③RecyclerView不關心item的動畫(交給ItemAnimator) 
④RecyclerView不關心item的布局(交給LayoutManager) 
本文為初探,所以先不詳解②③④這幾個內容,先記錄一下用法和遇到的坑,後續有時間再更新! 
先放一下效果圖,動畫效果在之前的部落格也有提到,可以參考,主要的是中間的RecyclerView,這裡主要講如何使用它以及遇到的一些問題

 

先添加依賴,版本無所謂:
 compile 'com.android.support:recyclerview-v7:23.1.1'
布局檔案如下,注意最後一個屬性可以不用理,這是用來協調動畫效果的(就是把上面圖片推上去的效果)

<android.support.v7.widget.RecyclerViewandroid:id="@+id/main_lv"android:layout_width="match_parent"android:layout_height="match_parent"app:layout_behavior="@string/appbar_scrolling_view_behavior"/> 

自訂一個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);  }}

解析:構造方法中擷取到資料來源以及根View,根View用來顯示SnackBar,如果不知道SnackBar,請看上一篇。然後建立一個內部類,繼承自RecyclerView.ViewHolder,用它來儲存每一個Item的全部控制項。接下來重寫幾個方法:getItemCount、onBindViewHolder以及onCreateViewHolder,他們分別是返回item個數,綁定ViewHolder以及建立ViewHolder,建立ViewHolder就是通過Item的布局檔案擷取每個控制項的執行個體,綁定ViewHolder就是把資料來源中的資料對應到Item的控制項中,因為我們的Item中還有刪除和修改按鈕,所以我們為它們添加點擊事件也是在onBindViewHolder中進行。接下來我們定義一個介面OnItemButtonClickListener並且定義兩個方法分別用來回調點擊刪除和修改,並建立介面的對象,在按鈕點擊的時候分別調用介面的兩個方法,並傳入點擊的item的position。
Activity部分代碼如下:

 rv = (RecyclerView) findViewById(R.id.main_lv);adapter = new MyAdapter(data, root);rv.setLayoutManager(new LinearLayoutManager(this,LinearLayoutManager.VERTICAL,false)); // 設定布局,否則無法正常使用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的執行個體之後,設定對應的布局管理器,然後通過setOnItemButtonClickListener設定回呼函數並實現,最後給RecyclerView設定adapter即可。
遇到的問題:
 ①剛開始忘記設定布局,RecyclerView不顯示出來
 ②因為我們在刪除item之後調用了notifyItemRemoved方法來顯示動畫並重新整理,在onBindViewHolder中,對刪除按鈕監聽之後調用介面方法的時候傳入了position,這個position是錯誤的,因為在綁定的時候固定了,如果資料來源的資料被刪除了,資料來源的size會變小,這個position就不再對應於資料來源的每一個項就會產生混亂,所以這裡應該傳入viewholder.getLayoutPosition()所得到的值,這個值是對應點擊而產生的。
 ③如果在RecyclerView的第一項前面插入一個item,並且調用notifyItmeInserted(0),是不會播放動畫效果的。

以上就是本文的全部內容,希望對大家的學習有所協助,也希望大家多多支援雲棲社區。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.