浅谈RecyclerView(完美替代ListView,GridView)

Android RecyclerView 是Android5.0推出来的,导入support-v7包即可使用。

个人体验来说,RecyclerView绝对是一款功能强大的控件。

首先总结下RecyclerView的特点:

1.支持不同方向,不同排版模式,实现多种展现数据的形式,涵盖了ListView,GridView,瀑布流等数据表现的形式

2.内部实现了回收机制,无需我们考虑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,列表项布局,数据源

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="添加一个数据"
  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一样 标准三样, 控件,数据源,适配器

private List<String> data;
private RecyclerView recyclerView;
private MyRecycleAdapter adapter; //自定义适配器,继承RecyclerView.Adapter

接着我们必须要自定义一个ViewHolder,这个ViewHolder 必须要继承 RecyclerView.ViewHolder

注意RecyclerView不再提供onItemClick事件监听,所以需要我们自己手工写监听事件的方法

private static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
 public ImageView imageView;
 public TextView textView;
 public ViewHolder(View itemView) {
  super(itemView);
  // 通常ViewHolder的构造,就是用于获取控件视图的
  imageView = (ImageView) itemView.findViewById(R.id.item_icon);
  textView = (TextView) itemView.findViewById(R.id.item_title);
  // TODO 后续处理点击事件的操作
  itemView.setOnClickListener(this);

 }
 @Override
 public void onClick(View v) {
  int position = getAdapterPosition();
  Context context = imageView.getContext();
  Toast.makeText(context,"显示第"+position+"个项",Toast.LENGTH_SHORT).show();
 }
 }

再让我们看自定义适配器,注意这里的参数是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中,首先要写一个构造方法,因为有数据源,所有构造方法里必然有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 这个属性

用于进行一个布局的设置,可以设置显示模式,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项数目
     GridLayoutManager.HORIZONTAL, //水平排列
     false
     );

3.瀑布流显示模式

// 3.瀑布流
 RecyclerView.LayoutManager layoutManager =
  new StaggeredGridLayoutManager(3, // 每行显示的item项数目
   StaggeredGridLayoutManager.VERTICAL); // 垂直排列

以上三种显示模式任意设置一种 就可以继续下面的代码

recyclerView.setLayoutManager(layoutManager);
 // 设置 RecyclerView的Adapter
 // 注意一定在设置了布局管理器之后调用
 adapter = new MyRecycleAdapter(data);
 recyclerView.setAdapter(adapter);

最后记得加上“添加一个数据”,“删除第一个数据”的按钮响应事件。

首先看一下以往我们对listview,gridview等等的删除某一项的操作

先在数据源中删除该位置的数据,然后刷新整个适配器,那么就可能会造成列表闪屏的问题,还有为了删除添加一个数据项而操作整个数据源的问题

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); //数据源先添加该数据
 notifyItemInserted(position); //在某个位置刷新即可
}
public void remove(ViewModel item) {
 int position = items.indexOf(item);
 items.remove(position); //数据源先删除该数据
 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);
 // 设置布局管理器
 // 支持 单列线性排列,支持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);
 // 设置 RecyclerView的Adapter
 // 注意一定在设置了布局管理器之后调用
 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,用于显示数据
 * 需要定义并且使用 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);
 }
 }
 /**
 * 创建自己的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的构造,就是用于获取控件视图的
  imageView = (ImageView) itemView.findViewById(R.id.item_icon);
  textView = (TextView) itemView.findViewById(R.id.item_title);
  // TODO 后续处理点击事件的操作
  itemView.setOnClickListener(this);
 }
 @Override
 public void onClick(View v) {
  int position = getAdapterPosition();
  Context context = imageView.getContext();
  Toast.makeText(context,"显示第"+position+"个项",Toast.LENGTH_SHORT).show();
 }
 }
}

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,同时也希望多多支持我们!

(0)

相关推荐

  • 解决EditText、ListView以及GridView同时使用,输入法自动跳出来的方法

    有一个Activity,里面有个EditText下面是个ListView,每次运行这个Activity,输入法都自动跳出来,太恼火了.因为这个ListView是显示数据库的,一开始怀疑数据库游标Cursor的问题,看来怀疑错了,就是EditText和ListView同时出现,导致输入法自动跳出来.事实上我们需要的是,点击EditText后,输入法才出来.解决方法有:方法1:在onCreate()里,this.getWindow().setSoftInputMode(WindowManager.L

  • asp.net获取ListView与gridview中当前行的行号

    本文实例讲述了asp.net获取ListView与gridview中当前行的行号.分享给大家供大家参考,具体如下: aspx中,在gridview/ListView中,有一模板列,就叫linkbutton,想单击它时,获取它所在行的索引值 ListView中: 第一种: <ItemTemplate> <tr> <td> <asp:LinkButton runat="server" ID="btnSelected" Text=

  • 关于Android中ListView嵌套GridView的问题

    问题 在Android开发中,遇到一个问题,是ListView嵌套GridView,需要点击整个ListView的Item进行跳转.但是在点击GridView区域时无法进行页面的跳转.这是因为GridView获得了焦点.导致点击无法跳转. 解决方法就是: 1.在Item最外层加上 android:descendantFocusability="blocksDescendants" 2.在Adapter中添加 holder.mGridView.setClickable(false); h

  • Android RecyclerView详解之实现 ListView GridView瀑布流效果

     什么是RecyclerView RecyclerView 是Google推出的最新的 替代ListView.GridView的组件,RecyclerView是用来显示大量数据的容器,并通过有限数量的子View,来提高滚动时的性能. 与ListView不同,RecyclerView 不再负责布局,而是专注于布局复用.布局主要通过 LayoutManager来管理,目前提供了3种常用的布局管理: LinearLayoutManager 线性布局管理器 (ListView效果) GridLayout

  • android ListView和GridView拖拽移位实现代码

    关于ListView拖拽移动位置,想必大家并不陌生,比较不错的软件都用到如此功能了.如:搜狐,网易,百度等,但是相比来说还是百度的用户体验较好,不偏心了,下面看几个示例:             首先说一下:拖拽ListView的item就不应该可以任意移动,只应该在ListView所在的范围内,而网易的你看看我都可以移动到状态栏了,虽然你做了处理,但是用户体验我个人感觉不好,在看看百度的,不仅控制了移动范围,更不错的百度的移动起来会时时的换位,看起来相当的形象,所以我认为这样相当的棒.说明一点

  • Android仿贴吧内容下的简单ListView嵌套GridView

    ListView嵌套GridView的简单实例 我的项目想实现一个listview里面的每个item都嵌套一个GridView,顶部还有主题等内容,如 总所周知,关于ListView嵌套GridView,最主要问题莫过于嵌套状态下滑动冲突问题,具体怎么解决,喜欢冗长无注释的代码的,请点击这里这篇文章跟其他的都大同小异了,不过在缺少注释的情况下,我发现了一点点小问题: /** * 创建日期:2017/3/21. * 说明:构造方法会根据你的SDK最低版本不同而要求不同,如18的至少必须重写前 *

  • Android开发之ListView、GridView 详解及示例代码

    ListView与GridView是Android开发中的常用控件,它们和Adapter配合使用能够实现很多界面效果.下面分别以实例说明ListView.GridView的用法.        1.ListView的Android开发实例 ListView 是android开发中最常用的控件之一,一般构成列表包括三个元素,ListView:用来展示列表的视图.Adapter:数据与视图连接的桥梁.Data:具体的数据包括字符串 .图片或者控件. 适配器一般有以下几种类型: ArrayAdapte

  • Android 中ScrollView嵌套GridView,ListView的实例

    Android 中ScrollView嵌套GridView,ListView的实例 在Android开发中,经常有一些UI需要进行固定style的动态布局,然而由于现在的UI都喜欢把一个界面拉的很长,所以我们很多情况下需要使用ScrollView来嵌套列表控件来实现UI.这样就导致了很多不顺心的问题. 问题一:列表控件显示不完全 原因是嵌套情况下,ScrollView不能正确的计算列表控件的高度. 有两种解决方案 方案一 在适配器赋值完成后代码动态计算列表的高度.这里贴出ListView的计算代

  • ListView嵌套GridView使用详解及注意事项

    MainActivity如下: 复制代码 代码如下: package cn.testlistviewandgridview; import java.util.ArrayList; import java.util.HashMap; import android.app.Activity; import android.os.Bundle; import android.widget.ListView; /** * Demo描述: * ListView嵌套GridView使用详解 * 即List

  • Android之ScrollView嵌套ListView和GridView冲突的解决方法

    那么里面的ScrollView高度计算就会出现问题.我们也就无法得到想要的效果.核心解决方案: 重写ListView或者GridView的OnMesure 方法. 复制代码 代码如下: public class MyListView extends ListView {        public MyListView(Context context) {                super(context);        }        public MyListView(Conte

随机推荐