android中RecyclerView的简单使用(一)

RecyclerView是什么?
RecyclerView是一种新的视图组,目标是为任何基于适配器的视图提供相似的渲染方式。它被作为ListView和GridView控件的继承者,在最新的support-V7版本中提供支持。

RecyclerView 是Android 版本中新添加的一个用来取代ListView的控件,它的灵活性与可替代性比listview更好。
RecyclerView与ListView原理是类似的:都是仅仅维护少量的View并且可以展示大量的数据集。

与传统ListView的区别:

RecyclerView与老前辈ListView的不同点,主要在于以下几个特性:
Adapter中的ViewHolder模式 - 对于ListView来说,通过创建ViewHolder来提升性能并不是必须的。因为ListView并没有严格的ViewHolder设计模式。但是在使用RecyclerView的时候,Adapter必须实现至少一个ViewHolder,必须遵循ViewHolder设计模式。


RecyclerView是android 5.0特性,用来取代listview和gridview的控件。

它的优点是:可以一个控件实现 listView  gridView 和瀑布流(推荐使用框架)
  解决了listView条目复用产生的图片错位的问题

它的缺点是:条目点击事件需要自己写
   分割线需要自己自定义编写
   5.0以下需要导入jar文件包


如何使用RecyclerView?

如果使用RecyclerView,你需要了解以下三个元素:

  • RecyclerView.Adapter
  • LayoutManager
  • ItemAnimator
RecyclerView.Adapter
确切的说,Adapter扮演着两个角色。一是,根据不同ViewType创建与之相应的的Item-Layout,二是,访问数据集合并将数据绑定到正确的View上。这就需要我们重写以下两个函数:

*public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) 创建Item视图,并返回相应的ViewHolder
*public void onBindViewHolder(VH holder, int position) 绑定数据到正确的Item视图上。

另外我们还需要重写另一个方法,像ListView-Adapter那样,同样地告诉RecyclerView-Adapter列表Items的总数:
public int getItemCount() 返回该Adapter所持有的Itme数量

ViewHolder的基本用法是用来存放View对象。Android团队很早之前就推荐使用“ViewHolder设计模式”,但实际上他们并没有把这种概念强加给开发者,而且也没有要求开发者在Adapter中必须使用ViewHolder pattern。那么现在对于这种新型的RecyclerView.Adapter,我们必须实现并使用它。

RecyclerView.LayoutManager
LayoutManager的职责是摆放Item的位置,并且负责决定何时回收和重用Item。必须为RecyclerView指定LayoutManager,否则会出现异常

目前SDK中提供了三种自带的LayoutManager:

LinearLayoutManager 水平或者垂直的Item视图。

GridLayoutManager 网格Item视图。

StaggeredGridLayoutManager 交错的网格Item视图。瀑布流

RecyclerView.ItemDecoration
通过设置recyclerView.addItemDecoration(new DividerDecoration(this));来改变Item之间的偏移量或者对Item进行装饰。
RecyclerView.ItemDecoration是一个抽象类,可以通过重写以下三个方法,来实现Item之间的偏移量或者装饰效果:
public void onDraw(Canvas c, RecyclerView parent) 装饰的绘制在Item条目绘制之前调用,所以这有可能被Item的内容所遮挡
public void onDrawOver(Canvas c, RecyclerView parent) 装饰的绘制在Item条目绘制之后调用,因此装饰将浮于Item之上
public void getItemOffsets(Rect outRect, int itemPosition, RecyclerView parent) 与padding或margin类似,LayoutManager在测量阶段会调用该方法,计算出每一个Item的正确尺寸并设置偏移量。
RecyclerView.ItemAnimator
ItemAnimator能够帮助Item实现独立的动画。
ItemAnimator作触发于以下三种事件:
某条数据被插入到数据集合中
从数据集合中移除某条数据 
更改数据集合中的某条数据
幸运的是,在Android中默认实现了一个DefaultItemAnimator,我们可以通过以下代码为Item增加动画效果:
recyclerView.setItemAnimator(new DefaultItemAnimator());

在之前的版本中,当时据集合发生改变时,我们通过调用.notifyDataSetChanged(),来刷新列表,因为这样做会触发列表的重绘,所以并不会出现任何动画效果,因此需要调用一些以notifyItem*()作为前缀的特殊方法,比如:

public final void notifyItemInserted(int position) 向指定位置插入Item
public final void notifyItemRemoved(int position) 移除指定位置Item
public final void notifyItemChanged(int position) 更新指定位置Item

这些就是RecyclerView的一些基础知识。下面贴一些代码,方便理解。

public class MainActivity extends Activity implements CallBack {
	private LinearLayout mLin;
	private RecyclerView recycle;
	private List<String> list=new ArrayList<String>();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initList();
        initView();
    }
	private void initView() {
		mLin=(LinearLayout) findViewById(R.id.mLin);
		recycle=new RecyclerView(this);
		recycle.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT));
		mLin.addView(recycle);
		LinearLayoutManager manager=new LinearLayoutManager(this);//线性布局管理器,支持横向和纵向
		manager.setOrientation(LinearLayoutManager.HORIZONTAL);
		recycle.setLayoutManager(manager);
		recycle.setHasFixedSize(true);//固定宽高
		recycle.setItemAnimator(new DefaultItemAnimator());//默认的条目添加动画
		recycle.setAdapter(new MyAdapter(this, list, this));
		
	}
	private void initList() {
		for(int i=0;i<30;i++){
			list.add("第"+(i+1)+"条数据");
		}
	}
	@Override
	public void call(int index) {
		Toast.makeText(this, list.get(index), 0).show();
	}

}
下面是adapter里面的代码:

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.VH>{
	private Context ctx;
	private List<String> list;
	private CallBack call;

	public MyAdapter(Context ctx,List<String> list,CallBack call) {
		this.ctx=ctx;
		this.list=list;
		this.call=call;
	}
	//内部类--》找ID
	class VH extends RecyclerView.ViewHolder{
		private TextView tv;
		private ImageView img;
		private View view;
		public VH(View arg0) {
			super(arg0);
			tv=(TextView) arg0.findViewById(R.id.tv);
			img=(ImageView) arg0.findViewById(R.id.img);
			view=arg0.findViewById(R.id.view);
		}
	}
	@Override
	public int getItemCount() {
		return list.size();
	}
	//arg0==holder   arg1:position
	@Override
	public void onBindViewHolder(VH arg0, final int arg1) {
		arg0.img.setImageResource(R.drawable.ic_launcher);
		arg0.tv.setText(list.get(arg1));
		arg0.view.setBackgroundColor(Color.BLACK);
		//自己写的条目点击事件
		arg0.tv.setOnClickListener(new OnClickListener() {

			@Override
			public void onClick(View arg0) {
				call.call(arg1);
			}
		});
	}
	@Override
	public VH onCreateViewHolder(ViewGroup arg0, int arg1) {
		View v=View.inflate(ctx, R.layout.item, null);
		return new VH(v);
	}
	public interface CallBack{
		public void call(int index);//index:是你要点击的那个条目对应的在list中的位置:position
	}
}

xml中的代码比较简单,就不贴了。

不足之处请多指教!!!


猜你喜欢

转载自blog.csdn.net/lzllzllhl/article/details/71673793