前言


上一篇的代码,也是基于这些封装的。

RV的封装,跟曾经的listView之类的封装,大同小异。

这里,从@devwiki 处,将代码搬过来。基本无改动

BaseHolder的优化


  1. 使ViewHolder仅仅用来缓存View。

  2. 加入SparseArray,使之来缓存View。

  3. 加入BaseHolder(View view)构造器,外部更方便控制View。

  4. 保留getContext()方法,方便获取Context对象。

  5. getView(resid)。简化itemView.findviewById()
/**
* from http://blog.devwiki.net/index.php/2016/07/17/Recycler-View-Adapter-ViewHolder-optimized.html
* 基础的ViewHolder</br>
* ViewHolder仅仅作View的缓存,不关心数据内容
* Created by DevWiki on 2016/5/17.
*/
public class BaseHolder extends RecyclerView.ViewHolder { private SparseArray<View> mViewArray; /**
* 构造ViewHolder (该方法涉及到parent,不经常使用)
* @param parent 父类容器
* @param resId 布局资源文件id
*/
public BaseHolder(ViewGroup parent, @LayoutRes int resId) {
super(LayoutInflater.from(parent.getContext()).inflate(resId, parent, false));
mViewArray = new SparseArray<>();
} /**
* 构造ViewHolder
* @param context
* @param resId 布局资源文件id
*/
public BaseHolder(Context context, @LayoutRes int resId) {
super(LayoutInflater.from(context).inflate(resId, null, false));
mViewArray = new SparseArray<>();
} /**
* 构建ViewHolder
* @param view 布局View
*/
public BaseHolder(View view) {
super(view);
mViewArray = new SparseArray<>();
} /**
* 获取布局中的View
* @param viewId view的Id
* @param <T> View的类型
* @return view
*/
public <T extends View> T getView(@IdRes int viewId){
View view = mViewArray.get(viewId);
if (view == null) {
view = itemView.findViewById(viewId);
mViewArray.put(viewId, view);
}
return (T) view;
} /**
* 获取Context实例
* @return context
*/
public Context getContext() {
return itemView.getContext();
}
}

Adapter部分的优化


Adapter拆分为两个抽象类:AbsAdapter与BaseAdapter,当中:

AbsAdapter:封装了和ViewHolder和HeaderView。FooterView相关的方法。

BaseAdapter:继承AbsAdapter。封装了数据相关的方法。

各自聚焦于不同的方面,方面日后扩展。

AbsAdapter的代码例如以下:

/**
* from http://blog.devwiki.net/index.php/2016/07/17/Recycler-View-Adapter-ViewHolder-optimized.html
* RecyclerView.Adapter的扩展,包括headerView/footerView等
* Created by DevWiki on 2016/7/13.
*/ public abstract class AbsAdapter<VH extends BaseHolder> extends RecyclerView.Adapter<BaseHolder> { private static final String TAG = "AbsAdapter"; public static final int VIEW_TYPE_HEADER = 1024;
public static final int VIEW_TYPE_FOOTER = 1025; protected View headerView;
protected View footerView; protected Context context; public AbsAdapter(Context context) {
this.context = context;
} @Override
public final BaseHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if (viewType == VIEW_TYPE_HEADER) {
return new BaseHolder(headerView);
} else if (viewType == VIEW_TYPE_FOOTER) {
return new BaseHolder(footerView);
} else {
return createCustomViewHolder(parent, viewType);
}
} /**
* 创建自己定义的ViewHolder
*
* @param parent 父类容器
* @param viewType view类型{@link #getItemViewType(int)}
* @return ViewHolder
*/
public abstract VH createCustomViewHolder(ViewGroup parent, int viewType); @Override
public final void onBindViewHolder(BaseHolder holder, int position) {
switch (holder.getItemViewType()) {
case VIEW_TYPE_HEADER:
case VIEW_TYPE_FOOTER:
break;
default:
bindCustomViewHolder((VH) holder, position);
break;
}
} @Override
public void onBindViewHolder(BaseHolder holder, int position, List<Object> payloads) {
super.onBindViewHolder(holder, position, payloads);
} /**
* 绑定自己定义的ViewHolder
*
* @param holder ViewHolder
* @param position 位置
*/
public abstract void bindCustomViewHolder(VH holder, int position); /**
* 加入HeaderView
*
* @param headerView 顶部View对象
*/
public void addHeaderView(View headerView) {
if (headerView == null) {
Log.w(TAG, "add the header view is null");
return ;
}
this.headerView = headerView;
notifyDataSetChanged();
} /**
* 移除HeaderView
*/
public void removeHeaderView() {
if (headerView != null) {
headerView = null;
notifyDataSetChanged();
}
} /**
* 加入FooterView
*
* @param footerView View对象
*/
public void addFooterView(View footerView) {
if (footerView == null) {
Log.w(TAG, "add the footer view is null");
return;
}
this.footerView = footerView;
notifyDataSetChanged();
} /**
* 移除FooterView
*/
public void removeFooterView() {
if (footerView != null) {
footerView = null;
notifyDataSetChanged();
}
} /**
* 获取附加View的数量,包括HeaderView和FooterView
*
* @return 数量
*/
public int getExtraViewCount() {
int extraViewCount = 0;
if (headerView != null) {
extraViewCount++;
}
if (footerView != null) {
extraViewCount++;
}
return extraViewCount;
} /**
* 获取顶部附加View数量,即HeaderView数量
* @return 数量
*/
public int getHeaderExtraViewCount() {
return headerView == null ? 0 : 1;
} /**
* 获取底部附加View数量,即FooterView数量
* @return 数量,0或1
*/
public int getFooterExtraViewCount() {
return footerView == null ? 0 : 1;
} @Override
public abstract long getItemId(int position); }

BaseAdapter的代码例如以下:

/**
* from http://blog.devwiki.net/index.php/2016/07/17/Recycler-View-Adapter-ViewHolder-optimized.html
* 基础的Adapter
*
* Created by DevWiki on 2016/7/13.
*/ public abstract class BaseAdapter<M, VH extends BaseHolder> extends AbsAdapter<VH> { private List<M> dataList; public BaseAdapter(Context context) {
super(context);
this.dataList = new ArrayList<>();
} public BaseAdapter(Context context, List<M> list) {
super(context);
this.dataList = new ArrayList<>();
this.dataList.addAll(list);
} /**
* 填充数据,此操作会清除原来的数据
*
* @param list 要填充的数据
* @return true:填充成功并调用刷新数据
*/
public boolean fillList(List<M> list) {
dataList.clear();
boolean result = dataList.addAll(list);
if (result) {
notifyDataSetChanged();
}
return result;
} /**
* 追加一条数据
*
* @param data 要追加的数据
* @return true:追加成功并刷新界面
*/
public boolean appendItem(M data) {
boolean result = dataList.add(data);
if (result) {
if (getHeaderExtraViewCount() == 0) {
notifyItemInserted(dataList.size() - 1);
} else {
notifyItemInserted(dataList.size());
}
}
return result;
} /**
* 追加集合数据
*
* @param list 要追加的集合数据
* @return 追加成功并刷新
*/
public boolean appendList(List<M> list) {
boolean result = dataList.addAll(list);
if (result) {
notifyDataSetChanged();
}
return result;
} /**
* 在最顶部前置数据
*
* @param data 要前置的数据
*/
public void proposeItem(M data) {
dataList.add(0, data);
if (getHeaderExtraViewCount() == 0) {
notifyItemInserted(0);
} else {
notifyItemInserted(getHeaderExtraViewCount());
}
} /**
* 在顶部前置数据集合
*
* @param list 要前置的数据集合
*/
public void proposeList(List<M> list) {
dataList.addAll(0, list);
notifyDataSetChanged();
} @Override
public long getItemId(int position) {
return position;
} @Override
public final int getItemViewType(int position) {
if (headerView != null && position == 0) {
return VIEW_TYPE_HEADER;
} else if (footerView != null && position == dataList.size() + getHeaderExtraViewCount()) {
return VIEW_TYPE_FOOTER;
} else {
return getCustomViewType(position);
}
} /**
* 获取自己定义View的类型
*
* @param position 位置
* @return View的类型
*/
public abstract int getCustomViewType(int position); @Override
public int getItemCount() {
return dataList.size() + getExtraViewCount();
} /**
* 依据位置获取一条数据
*
* @param position View的位置
* @return 数据
*/
public M getItem(int position) {
if (headerView != null && position == 0
|| position >= dataList.size() + getHeaderExtraViewCount()) {
return null;
}
return headerView == null ? dataList.get(position) : dataList.get(position - 1);
} /**
* 依据ViewHolder获取数据
*
* @param holder ViewHolder
* @return 数据
*/
public M getItem(VH holder) {
return getItem(holder.getAdapterPosition());
} public void updateItem(M data) {
int index = dataList.indexOf(data);
if (index < 0) {
return;
}
dataList.set(index, data);
if (headerView == null) {
notifyItemChanged(index);
} else {
notifyItemChanged(index + 1);
}
} /**
* 移除一条数据
*
* @param position 位置
*/
public void removeItem(int position) {
if (headerView == null) {
dataList.remove(position);
} else {
dataList.remove(position - 1);
}
notifyItemRemoved(position);
} /**
* 移除一条数据
*
* @param data 要移除的数据
*/
public void removeItem(M data) {
int index = dataList.indexOf(data);
if (index < 0) {
return;
}
dataList.remove(index);
if (headerView == null) {
notifyItemRemoved(index);
} else {
notifyItemRemoved(index + 1);
}
}
}

參考


http://blog.devwiki.net/index.php/2016/07/17/Recycler-View-Adapter-ViewHolder-optimized.html 《RecyclerView的ViewHolder和Adapter的封装优化》

从头開始学 RecyclerView(三) 封装简化的更多相关文章

  1. 从头開始学 RecyclerView(六) LayoutManager

    前言 在前面的文章中.每一个演示样例,都使用了LayoutManager,毕竟它是RecyclerView必不可少的一部分. LayoutManager,顾名思义,就是『布局管理器』. 使用例如以下代 ...

  2. [php learn] php 从头開始学习1

    前言:大概在2006年的时候,学习过一段时间的php.而且当时做了一个下载的站点,后来因为读研究生阶段用的是java.j2ee相关,所以php就搁浅掉了,php这些年也发生了非常大的变化,最大一个变化 ...

  3. 送给刚刚開始学cocos2d-x引擎 移植Android的同学

    刚刚開始学cocos2-x,不过依照教程把已经安了一般Android的开发环境的eclipse又一次升级到安装好cdt和ndk就花了我几十小时,差点都要放弃了. 參考博客 http://blog.cs ...

  4. 从零開始学Swift之Hello World进化版

    上节课,也就是昨晚啦,我们学习到从零開始学Swift之Hello World.那一节仅仅有一句代码,大家会认为不够过瘾. 那么这节课,就给大家来多点瘾货吧! 先上图! //var 代表变量的类型, s ...

  5. 关东升的《从零開始学Swift》即将出版

    大家好: 苹果2015WWDC大会公布了Swift2.0,它较之前的版本号Swift1.x有非常大的变化.所以我即将出版<从零開始学Swift><从零開始学Swift>将在&l ...

  6. 《PHP 5.5从零開始学(视频教学版)》内容简单介绍、文件夹

    <PHP 5.5从零開始学(视频教学版)>当当网购买地址: http://product.dangdang.com/23586810.html <PHP 5.5从零開始学(视频教学版 ...

  7. 从零開始学android&lt;数据存储(1)SharedPreferences属性文件.三十五.&gt;

    在android中有五种保存数据的方法.各自是: Shared Preferences Store private primitive data in key-value pairs. 相应属性的键值 ...

  8. 【高德地图API】从零開始学高德JS API(五)路线规划——驾车|公交|步行

    先来看两个问题:路线规划与导航有什么差别?步行导航与驾车导航有什么差别? 回答: 1.路线规划,指的是为用户提供3条路线推荐.[高德]在提供路线规划的时候,会提供用户自己定义路线规划功能,这是别家没有 ...

  9. 第13章、布局Layouts之RelativeLayout相对布局(从零開始学Android)

    RelativeLayout相对布局 RelativeLayout是一种相对布局,控件的位置是依照相对位置来计算的,后一个控件在什么位置依赖于前一个控件的基本位置,是布局最经常使用,也是最灵活的一种布 ...

随机推荐

  1. Orchard之模版开发

    生成新模版之后(参看:Orchard之生成新模板),紧接着就是模版开发了. 一:开发必备之 Shape Tracing 到了这一步,非常依赖一个工具,当然,它也是 Orchard 项目本身的一个 Mo ...

  2. 开启VIM的Python支持

    开启VIM的Python支持 2015年01月03日 02:57:58 forlong401 阅读数:16294更多 个人分类: VIPython   http://www.tuicool.com/a ...

  3. 内存数据库-H2简介与实践

    一.H2数据库介绍 H2数据库地址:http://www.h2database.com/html/main.html H2是一个开源的嵌入式(非嵌入式设备)数据库引擎,它是一个用Java开发的类库,可 ...

  4. [转]mysql 一个表两列的值交换

    FROM : http://bbs.csdn.net/topics/380025779 mysql> select * from test1 +------+-------+-------+ | ...

  5. OpenGL教程(25) skybox

    原帖地址:http://ogldev.atspace.co.uk/www/tutorial25/tutorial25.html Background A skybox is a technique t ...

  6. java的关于流程结构做的几个案例

    最近在学习中,做了一个java的几个案例,主要是九九乘法口诀,实心菱形和空心菱形的算法,模拟彩票程序以及BMI的测试标准等小案例. 一:九九乘法表 /** * 九九乘法口诀 */ public sta ...

  7. UML图与软件开发过程那点关系

    首先,软工文档, 软工文档,也就是计划,设计,描述,使用软件的一些文件,它最大的特点就是固定不变,用来给不同的人和计算机来阅读.在期间,文档起到了桥梁的作用,看这张图很形象: 在这里在看一下国家统一规 ...

  8. NodeJS错误-throw er; // Unhandled 'error' event

    第一眼看以为Express版本出现问题,因为本地已经存在另外一个运行的Node项目,端口重复,修改一下端口号即可,错误提示如下: events.js:85 throw er; // Unhandled ...

  9. vue组件的hover事件模拟、给第三方组件绑定事件不生效问题

    1.vue里面实现hover效果基本需要用事件模拟 <div @mouseover="overShow" @mouseout="outHide"> ...

  10. Oracle学习笔记(6)——函数

    函数的作用 方便数据的统计 处理查询结果 函数的分类 Oracle内置的系统函数 数值函数 四舍五入                                             ROUND ...