深入RecyclerView-为什么要使用ItemDecoration
Part 1:不要用view做分割线

首先,什么是ItemDecoration?来看看官网是如何解释的。
ItemDecoration允许从adapter的数据集合中为特定的item视图添加特性的绘制以及布局间隔。它可以用来实现item之间的分割线,高亮,分组边界等。
我们不能简单的把ItemDecoration看成一个名字响亮的分割线。它比divider要多很多内容。一个divider只能绘制在item之间,但是ItemDecoration可以绘制在item的四边。ItemDecoration为decoration的测量和绘制提供了全方位的控制。一个decoration可以是一条分割线,也可以仅仅是一个间隔(inset)。
但不幸的是,绝大多数android开发者都没有使用item decoration。在这个分为三部分的系列文章中,我们将了解ItemDecoration的强大之处。
第一部分: 不要添加view来做分割线— 使用 ItemDecoration
第二部分: 不要使用padding来做间隔 —使用 ItemDecoration
第三部分: 在GridLayoutManager中高效的绘制decorations
本文是第一部分。
不要用view做分割线 —会影响性能
我曾看到一些开发者在为RecyclerView添加divider的时候采用了一些捷径。原因很简单,ListView原生支持divider,可以直接在xml中设置divider。
<ListView
android:id="@+id/activity_home_list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:divider="@android:color/black"
android:dividerHeight="8dp"/>
但是到了RecyclerView,就再也不能直接添加divider了。需要添加一个绘制divider的ItemDecoration。但是开发者发现它很麻烦,于是直接把divider添加到(item的)view上,而不是使用ItemDecoration。
<LinearLayout android:orientation="vertical">
<LinearLayout android:orientation="horizontal">
<ImageView />
<TextView />
</LinearLayout>
<View
android:width="match_parent"
android:height="1dp"
android:background="#333" />
</LinearLayout>
每当我们走捷径的时候,都有可能会产生副作用。而这里的副作用是可能影响性能。
当在布局中添加了一个divider的时候,我们增加了view的个数。我们都知道view的数目越少会得到越好的性能。有时候增加一个view来实现divider还会增加布局的层级。比如上面的例子中,我们不仅仅增加了一个view,还增加了一个包含它们的 linear layout。为了一个divider而创建了额外的布局。
不要用view做分割线 —会带来副作用
因为divider是view的一部分,所以在item 动画期间,divider也会一起跟着动画。如下图:

显然divider不应该随着item一起做动画。而是和item分开,像这样才是对的:

不要用view做分割线— 缺乏灵活性
如果divider是(item的)view的一部分,那么你就无法控制它。你唯一能控制的就是根据item的position改变divider的可见状态。 而item decoration就灵活多了。

In the above image for the last item in the group divider fills the entire width. Other dividers have a margin of 56dp to their left side. Here is the ItemDecorator’s onDraw code.
在上图中,group最后一个item的divider充满了整个宽度。其它的divider都有一个56dp的左边距。这是这个ItemDecorator的onDraw代码:
@Override
public void onDraw(Canvas canvas, RecyclerView parent, RecyclerView.State state) {
canvas.save();
final int leftWithMargin = convertDpToPixel(56);
final int right = parent.getWidth(); final int childCount = parent.getChildCount();
for (int i = 0; i < childCount; i++) {
final View child = parent.getChildAt(i);
int adapterPosition = parent.getChildAdapterPosition(child);
left = (adapterPosition == lastPosition) ? 0 : leftWithMargin;
parent.getDecoratedBoundsWithMargins(child, mBounds);
final int bottom = mBounds.bottom + Math.round(ViewCompat.getTranslationY(child));
final int top = bottom - mDivider.getIntrinsicHeight();
mDivider.setBounds(left, top, right, bottom);
mDivider.draw(canvas);
}
canvas.restore();
}
不要用view做分割线—使用 ItemDecoration
写一个自己的ItemDecoration其实非常简单。你只需要创建一个继承了ItemDecoration的类就可以了。重写 getItemOffsets() 和 onDraw() 方法。具体实现可以参考 这个 示例。
而 25.0.0版本的支持库中,我们有一个新的类 “DividerItemDecoration”。这个类直接实现了divider。
DividerItemDecoration decoration = new DividerItemDecoration(getApplicationContext(), VERTICAL);
recyclerView.addItemDecoration(decoration);
提示
一个RecyclerView可以添加多个ItemDecoration。发挥头脑风暴的时候到了。
所有decoration都在item绘制之前绘制。如果你想让decoration在view之后绘制,重写onDrawOver() 而不是onDraw() 。
所以下次想为RecyclerView添加分割线的时候,别使用在item布局添加view这种方式了,使用ItemDecoration。
深入RecyclerView-为什么要使用ItemDecoration的更多相关文章
- 深入理解 RecyclerView 系列之:ItemDecoration
		
https://blog.piasy.com/2016/03/26/Insight-Android-RecyclerView-ItemDecoration/?utm_source=tuicool&am ...
 - 小甜点,RecyclerView 之 ItemDecoration 讲解及高级特性实践
		
本篇文章摘自微信公众号 guolin_blog (郭霖)独家发布 毫无疑问,RecyclerView 是现在 Android 世界中最重要的系统组件之一,它的出现就是为了高效代替 ListView 和 ...
 - Android RecyclerView体验(一)- 简介
		
在网上关于RecyclerView的基本使用方式已经有了比较详细介绍,而且其设计结构也类似于ListView,所以本文将不重点介绍如何使用,在文末的引用中都可以相关内容.这里主要是介绍Recycler ...
 - Android RecyclerView完全解析
		
RecyclerView完全解析 (一) 前言 话说RecyclerView已经面市很久,也在很多应用中得到广泛的使用,在整个开发者圈子里面也拥有很不错的口碑,那说明RecyclerView拥有比Li ...
 - RecyclerView源码分析(一)--整体设计
		
RecyclerView这个控件出来已经有一段时间了,如果看这篇文章的你,还没有使用过这个控件.那请先去学习怎样使用.不然看也白看.这里奉上一些关于介绍RecyclerView使用方法的优秀博客: 鸿 ...
 - 【转载】RecyclerView源码解析
		
原文地址:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2016/0307/4032.html 概述 之前面试的时候经常有人问是否用过Re ...
 - 自定义RecyclerView.ItemDecoration,实现RecyclerView的分割线效果
		
[转] 原文 自定义RecyclerView.ItemDecoration,实现RecyclerView的分割线效果 字数1598 阅读302 评论2 喜欢23 1.背景 RecyclerView ...
 - RecyclerView.ItemDecoration
		
decoration 英文意思: 英[ˌdekəˈreɪʃn] 美[ˌdɛkəˈreʃən] n. 装饰品; 装饰,装潢; 装饰图案,装饰风格; 奖章; [例句]The decoration and ...
 - android RecyclerView (二) ItemDecoration 详解
		
RecyclerView 已经推出了一年多了,日常开发中也已经彻底从 ListView 迁移到了 RecyclerView,但前两天有人在一个安卓群里面问了个关于最顶上的 item view 加蒙层的 ...
 - RecyclerView.ItemDecoration 间隔线
		
内容已更新到:https://www.cnblogs.com/baiqiantao/p/19762fb101659e8f4c1cea53e7acb446.html 目录一个通用分割线ItemDecor ...
 
随机推荐
- DP练习 巡逻
			
国庆这天五一大道上人头攒动,这是因为大家都准备从五一广场上那个大屏幕观看新中国60周年的国庆阅兵式!这虽然是一件很喜庆的事情,可却让CS市的警察局长伤透了脑筋的,因为人潮拥挤是很容易发生安全事故的. ...
 - ACM -- 算法小结(一)利用数组存放实现排序
			
利用数组存放实现排序 hodj1425 321MS 2011/08 题意:输入n个数字,要求输出从大到小排序的前m个数 解题技巧:利用大数存储在数组后面,小数存储在前面,倒序输出完成从大 ...
 - timeago.js-时间显示插件
			
注意事项: 1. 时间格式 = “2018-03-02 17:13:00”时,动态获取的时间无法通过 拼接字符串的方法 添加到 dom元素的 datetime属性上,结果为 <div class ...
 - 单端通用ISM频段接收器 Si4313
			
Si4313芯片是单端通用ISM频段接收器,工作频率为240-960MHz,可编程接收频率带宽为2.6-260kHz,接收灵敏度为-118dBm,数据速率为0.2-128kb/s,采用FSK.GFSK ...
 - 用matplotlib绘制带误差的条形图及中英文字体设置
			
#!/usr/bin/env python3 ## 以下是一个带误差条的条形图的例子,演示了误差条形图的绘制及中英文字体设置 import numpy as np import matplotlib ...
 - Druid 连接泄漏监测
			
当程序存在缺陷时,申请的连接忘记关闭,这时候,就存在连接泄漏了.Druid提供了RemoveAbandanded相关配置,用来关闭长时间不使用的连接.例如: 配置 <bean id=" ...
 - Linux Kernel 4.9 & BBR
			
https://www.mxgw.info/t/linux-kernel-4-9-bbr.html?utm_source=tuicool&utm_medium=referral
 - gitignore不起作用解决的方法
			
前面有文章介绍了使用gitignore文件的方法,该文件表示过滤规则,可是对已经增加版本号库的文件不能生效,因此须要利用命令将想要忽略的文件从版本号库中删除,比方说.我们对androidproject ...
 - Python任务调度模块 – APScheduler。动态修改调度时间间隔
			
APScheduler可以把调度任务放到内存里,也可以把任务放到数据库里,那么如何交互式修改定时任务的执行时间间隔或者下次执行时间呢? 方案一:把定时任务放到数据库里,修改数据库里任务的调度时间 方案 ...
 - zedboard--基于zedboard的demo系统的boa服务器搭建(二十一)
			
zedboard提供的demo系统很迷你,但是也能移植嵌入式Web服务器的.这里就移植boa服务器. 1.下载Boa服务器源代码(安装好了交叉编译器) http://www.boa.org/,选择最后 ...