淺談RecyclerView(完美替代ListView,GridView)
Android RecyclerView 是Android5.0推出來的,導入support-v7包即可使用。
個人體驗來說,RecyclerView絕對是一款功能強大的控件。
首先總結(jié)下RecyclerView的特點:
1.支持不同方向,不同排版模式,實現(xiàn)多種展現(xiàn)數(shù)據(jù)的形式,涵蓋了ListView,GridView,瀑布流等數(shù)據(jù)表現(xiàn)的形式
2.內(nèi)部實現(xiàn)了回收機制,無需我們考慮View的復用情況
3.取消了onItemClick等點擊事件,需要自己手動去寫
那么讓我們通過一些Demo來了解RecyclerView的基本使用
android studio
build.gradle文件中 dependencies中添加
compile 'com.android.support:recyclerview-v7:22.+'
首先,要導入support-v7 包
import android.support.v7.widget.RecyclerView;
RecyclerView和ListView的使用一樣,都需要有對應的Adapter,列表項布局,數(shù)據(jù)源
1.先寫主Activity布局
可以看到RecyclerView的標簽
<android.support.v7.widget.RecyclerView>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" tools:context="com.xqx.superapp.app.Android5Activity"> <Button android:text="添加一個數(shù)據(jù)" android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="btnAddItem" /> <Button android:text="刪除第一個" android:onClick="btnRemoveItem" android:layout_width="wrap_content" android:layout_height="wrap_content"/> <android.support.v7.widget.RecyclerView android:id="@+id/recycle_view" android:layout_width="match_parent" android:layout_height="match_parent" > </android.support.v7.widget.RecyclerView> </LinearLayout>
菜單項布局,標準的上面圖片,下面文字
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:gravity="center" android:layout_width="match_parent" android:layout_height="match_parent"> <ImageView android:id="@+id/item_icon" android:src="@mipmap/machao_moqi" android:layout_width="wrap_content" android:layout_height="wrap_content"/> <TextView android:id="@+id/item_title" android:text="名稱" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout>
2.接下來就看Activity代碼了
首先看成員變量,與ListView,GridView一樣 標準三樣, 控件,數(shù)據(jù)源,適配器
private List<String> data; private RecyclerView recyclerView; private MyRecycleAdapter adapter; //自定義適配器,繼承RecyclerView.Adapter
接著我們必須要自定義一個ViewHolder,這個ViewHolder 必須要繼承 RecyclerView.ViewHolder
注意RecyclerView不再提供onItemClick事件監(jiān)聽,所以需要我們自己手工寫監(jiān)聽事件的方法
private static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public ImageView imageView;
public TextView textView;
public ViewHolder(View itemView) {
super(itemView);
// 通常ViewHolder的構(gòu)造,就是用于獲取控件視圖的
imageView = (ImageView) itemView.findViewById(R.id.item_icon);
textView = (TextView) itemView.findViewById(R.id.item_title);
// TODO 后續(xù)處理點擊事件的操作
itemView.setOnClickListener(this);
}
@Override
public void onClick(View v) {
int position = getAdapterPosition();
Context context = imageView.getContext();
Toast.makeText(context,"顯示第"+position+"個項",Toast.LENGTH_SHORT).show();
}
}
再讓我們看自定義適配器,注意這里的參數(shù)是ViewHolder,這個ViewHodler是我們自己的,不要導入v7包下的ViewHolder,
之后要重寫三個方法
private class MyRecycleAdapter extends RecyclerView.Adapter<ViewHolder>{
@Override
public ViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
return null;
}
@Override
public void onBindViewHolder(ViewHolder viewHolder, int i) {
}
@Override
public int getItemCount() {
return 0;
}
}
在自定義適配器MyRecycleAdapter中,首先要寫一個構(gòu)造方法,因為有數(shù)據(jù)源,所有構(gòu)造方法里必然有List
private List<String> strings;
public MyRecycleAdapter(List<String> strings) {
this.strings = strings;
}
然后就要重寫三個方法了,
@Override
public int getItemCount() {
int ret = 0;
if (strings != null) {
ret = strings.size();
}
return ret;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
ViewHolder ret = null;
// 不需要檢查是否復用,因為只要進入此方法,必然沒有復用
// 因為RecyclerView 通過Holder檢查復用
View v = LayoutInflater.from(Android5Activity.this).inflate(R.layout.item_recycler, viewGroup, false);
ret = new ViewHolder(v);
return ret;
}
@Override
public void onBindViewHolder(ViewHolder viewHolder, int i) {
// 1.這里進行圖片的加載
viewHolder.textView.setText(strings.get(i));
int resId = R.mipmap.ic_launcher;
int index = i%5;
switch (index){
case 0:
resId = R.mipmap.a11;
break;
case 1:
resId = R.mipmap.a33;
break;
case 2:
resId = R.mipmap.a22;
break;
}
viewHolder.imageView.setImageResource(resId);
}
完成自定義適配器和自定義ViewHolder的代碼 就要進行RecyclerView的使用了
首先 要了解 RecyclerView.LayoutManager 這個屬性
用于進行一個布局的設(shè)置,可以設(shè)置顯示模式,ListView或者GridView或者瀑布流
1.ListView顯示模式
// 1.線性布局
LinearLayoutManager layoutManager =
new LinearLayoutManager(this, // 上下文
LinearLayout.VERTICAL, //垂直布局,
false);


2.GridView顯示模式
// 2.Grid布局
RecyclerView.LayoutManager layoutManager =
new GridLayoutManager(this,
2, // 每行顯示item項數(shù)目
GridLayoutManager.HORIZONTAL, //水平排列
false
);


3.瀑布流顯示模式
// 3.瀑布流 RecyclerView.LayoutManager layoutManager = new StaggeredGridLayoutManager(3, // 每行顯示的item項數(shù)目 StaggeredGridLayoutManager.VERTICAL); // 垂直排列

以上三種顯示模式任意設(shè)置一種 就可以繼續(xù)下面的代碼
recyclerView.setLayoutManager(layoutManager); // 設(shè)置 RecyclerView的Adapter // 注意一定在設(shè)置了布局管理器之后調(diào)用 adapter = new MyRecycleAdapter(data); recyclerView.setAdapter(adapter);
最后記得加上“添加一個數(shù)據(jù)”,“刪除第一個數(shù)據(jù)”的按鈕響應事件。
首先看一下以往我們對listview,gridview等等的刪除某一項的操作
先在數(shù)據(jù)源中刪除該位置的數(shù)據(jù),然后刷新整個適配器,那么就可能會造成列表閃屏的問題,還有為了刪除添加一個數(shù)據(jù)項而操作整個數(shù)據(jù)源的問題
public void btnAddItem(View view) {
data.add(0,"Time:"+System.currentTimeMillis());
adapter.notifyDataSetChanged();
}
public void btnRemoveItem(View view) {
if (!data.isEmpty()) {
data.remove(0);
}
adapter.notifyItemRemoved(0);
}
而RecyclerView為我們提供了一些新的實用的方法:
public void add(ViewModel item, int position) {
items.add(position, item); //數(shù)據(jù)源先添加該數(shù)據(jù)
notifyItemInserted(position); //在某個位置刷新即可
}
public void remove(ViewModel item) {
int position = items.indexOf(item);
items.remove(position); //數(shù)據(jù)源先刪除該數(shù)據(jù)
notifyItemRemoved(position); //在某個位置刪除即可
}
完整代碼:
package com.xqx.superapp.app;
import android.app.Activity;
import android.content.Context;
import android.support.v7.app.ActionBarActivity;
import android.os.Bundle;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
import android.util.Log;
import android.view.*;
import android.widget.*;
import java.util.LinkedList;
import java.util.List;
public class Android5Activity extends Activity {
private List<String> data;
private RecyclerView recyclerView;
private MyRecycleAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_android5);
data = new LinkedList<String>();
recyclerView = (RecyclerView) findViewById(R.id.recycle_view);
// 設(shè)置布局管理器
// 支持 單列線性排列,支持GridView模式,瀑布流模式
// 1.線性布局
LinearLayoutManager layoutManager =
new LinearLayoutManager(this, // 上下文 LinearLayout.VERTICAL, //垂直布局,
false);
// // 2.Grid布局
// RecyclerView.LayoutManager layoutManager =
// new GridLayoutManager(this,
// 2,
// GridLayoutManager.HORIZONTAL,
// false
// );
//
// // 3.瀑布流
// RecyclerView.LayoutManager layoutManager =
// new StaggeredGridLayoutManager(3,
// StaggeredGridLayoutManager.VERTICAL);
recyclerView.setLayoutManager(layoutManager);
// 設(shè)置 RecyclerView的Adapter
// 注意一定在設(shè)置了布局管理器之后調(diào)用
adapter = new MyRecycleAdapter(data);
recyclerView.setAdapter(adapter);
}
public void btnAddItem(View view) {
data.add(0,"Time:"+System.currentTimeMillis());
adapter.notifyDataSetChanged();
}
public void btnRemoveItem(View view) {
if (!data.isEmpty()) {
data.remove(0);
}
adapter.notifyItemRemoved(0);
}
/**
* 繼承RecyclerView.Adapter,用于顯示數(shù)據(jù)
* 需要定義并且使用 ViewHolder ,必須要使用
*/
private class MyRecycleAdapter extends RecyclerView.Adapter<ViewHolder>{
private List<String> strings;
public MyRecycleAdapter(List<String> strings) {
this.strings = strings;
}
@Override
public int getItemCount() {
int ret = 0;
if (strings != null) {
ret = strings.size();
}
return ret;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
ViewHolder ret = null;
// 不需要檢查是否復用,因為只要進入此方法,必然沒有復用
// 因為RecyclerView 通過Holder檢查復用
View v = LayoutInflater.from(Android5Activity.this).inflate(R.layout.item_recycler, viewGroup, false);
ret = new ViewHolder(v);
return ret;
}
@Override
public void onBindViewHolder(ViewHolder viewHolder, int i) {
viewHolder.textView.setText(strings.get(i));
int resId = R.mipmap.ic_launcher;
int index = i%5;
switch (index){
case 0:
resId = R.mipmap.a11;
break;
case 1:
resId = R.mipmap.a33;
break;
case 2:
resId = R.mipmap.a22;
break;
}
viewHolder.imageView.setImageResource(resId);
}
}
/**
* 創(chuàng)建自己的ViewHolder ,必須要繼承RecyclerView.ViewHolder
*/
private static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public ImageView imageView;
public TextView textView;
public ViewHolder(View itemView) {
super(itemView);
// 通常ViewHolder的構(gòu)造,就是用于獲取控件視圖的
imageView = (ImageView) itemView.findViewById(R.id.item_icon);
textView = (TextView) itemView.findViewById(R.id.item_title);
// TODO 后續(xù)處理點擊事件的操作
itemView.setOnClickListener(this);
}
@Override
public void onClick(View v) {
int position = getAdapterPosition();
Context context = imageView.getContext();
Toast.makeText(context,"顯示第"+position+"個項",Toast.LENGTH_SHORT).show();
}
}
}
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,同時也希望多多支持腳本之家!
- Android中實現(xiàn)淘寶購物車RecyclerView或LIstView的嵌套選擇的邏輯
- Android添加圖片到ListView或者RecyclerView顯示
- Android RecyclerView詳解之實現(xiàn) ListView GridView瀑布流效果
- 將替代ListView的RecyclerView 的使用詳解(一)
- Android App開發(fā)中使用RecyclerView替代ListView的實踐
- Android仿XListView支持下拉刷新和上劃加載更多的自定義RecyclerView
- 學習Android Material Design(RecyclerView代替ListView)
- RecyclerView使用詳解(代替ListView)
相關(guān)文章
Android中使用PagerSlidingTabStrip實現(xiàn)導航標題的示例
本篇文章主要介紹了Android中使用PagerSlidingTabStrip實現(xiàn)導航標題的示例,具有一定的參考價值,有興趣的可以了解一下。2017-01-01
Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼
這篇文章主要介紹了Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼的實例代碼,具有很好的參考價值,希望對大家有所幫助,一起跟隨小編過來看看吧2018-05-05
Android 實現(xiàn)調(diào)用系統(tǒng)照相機拍照和錄像的功能
這篇文章主要介紹了Android 實現(xiàn)調(diào)用系統(tǒng)照相機拍照和錄像的功能的相關(guān)資料,需要的朋友可以參考下2016-11-11

