如何让 RecyclerView 平滑滚动?
Posted
技术标签:
【中文标题】如何让 RecyclerView 平滑滚动?【英文标题】:How to make RecyclerView scroll smoothly? 【发布时间】:2015-09-23 19:13:13 【问题描述】:这更像是一个通用问题,但经过大量搜索和尝试后,我无法理解为什么这很难实现。 This 是我能找到但仍然无法实施的最接近的答案。
具体来说,我将 RecyclerView 与 GridLayoutManager 一起使用。我想要的只是平滑滚动的网格布局(如默认的画廊应用程序),没什么花哨的,但网格布局管理器的默认实现以“生涩”的方式滚动视图。我尝试实现上述链接中的方法,但没有成功。
我也尝试实现 LinearSmoothScroller 但我不确定如何实现 computeScrollVectorForPosition 方法。 computeScrollVectorForPosition 上的Google's documentation 字面上有0 个单词。
我找到了this 3 part tutorial,但帮助甚微。所以,我想问的是:是否有某种模板代码可以在 LinearSmoothScroller 中实现,或者通过扩展 RecyclerView.SmoothScroller 来实现平滑滚动?即使代码取决于网格布局中每行的项目数和项目数,也必须有一些方法可以轻松完成。我在这里错过了什么吗?
【问题讨论】:
“我在这里遗漏了什么吗?” -- 或许您应该使用 Traceview、StrictMode
等工具来确定您现有代码为何“生涩”。
@commonsWare 所以你的意思是 GridLayoutManager 的默认实现应该平滑地滚动 gridview ?对不起,如果我听起来很菜鸟,我对 android 编程很陌生。
"所以你的意思是 GridLayoutManager 的默认实现应该平滑地滚动 gridview ?" ——是的,至少对于我对“顺利”的定义。 Android 中“生涩”滚动的典型来源是应用程序在主应用程序线程更新 UI 上花费了太多时间。对于RecyclerView
,这意味着在onBindViewHolder()
或onCreateViewHolder()
中花费太多时间。这些每个都需要在亚毫秒时间内返回,这意味着您不能在其中执行磁盘 I/O 或网络 I/O。
@CommonsWare 谢谢你的回答,我想我找到了问题所在。我在 onBindViewHolder() 上使用holder.photo.setImageBitmap(BitmapFactory.decodeFile(itemList.get(position).getAbsolutePath()));
,每个图像文件大约 100kb,列表中有十几个图像。我会尝试更好地实现它。
【参考方案1】:
在 Activity
或 Fragment
中声明 RecyclerView
的任何位置添加此内容
RecyclerView mRecyclerview = (RecyclerView) findViewById(...);
mRecyclerview.setNestedScrollingEnabled(false);
setNestedScrollview(false)
为您完成工作。
【讨论】:
【参考方案2】:Android 中“生涩”滚动的典型原因是应用程序在主应用程序线程更新 UI 上花费了太多时间。对于RecyclerView
,这意味着在onBindViewHolder()
或onCreateViewHolder()
中花费太多时间。这些每个都需要在亚毫秒时间内返回,这意味着您不能在其中执行磁盘 I/O 或网络 I/O。
认为我找到了问题所在。我正在使用 holder.photo.setImageBitmap(BitmapFactory.decodeFile(itemList.get(position).getAbsolutePath()));在 onBindViewHolder() 上,每个图像文件大约 100kb,列表中有大约十几个图像
是的,这将在主应用程序线程上执行磁盘 I/O 和图像解码。这将慢到足以导致 UI 卡顿(即“生涩”滚动)。
考虑使用图像加载库,例如 Picasso 或 Universal Image Loader,因为它们可以从位图中异步填充您的 ImageView
。
This sample app 使用 Universal Image Loader 来帮助填充 RecyclerView
(使用 GridLayoutManager
)的内容,数据源是设备上的可用视频。
【讨论】:
谢谢,我会检查并实施。同时,使用this 简单的解决方案而不是使用单独的库怎么样? @rockfight:因为这与您的问题无关。该页面的重点是内存效率,而不是主应用程序线程时间。 UIL 在图像驻留在 sd 卡中时不起作用(文件路径以/storage
开头。我为此使用了 Glide
,它为我的问题提供了一个完美的解决方案
这对我有用。我在图像大尺寸时面临同样的问题。当我们使用Bitmap
在ImageView
中设置图像时,它就像@CommonsWare 答案一样工作。
@CommonsWare,在我的应用程序中,recyclerview 滚动不流畅,因为我在 onBindViewHolder 内膨胀视图,有没有其他方法可以添加没有布局膨胀器的视图。【参考方案3】:
我刚遇到这个问题,禁用嵌套滚动修复了它。 这样做:
yourRecyclerview.setNestedScrollingEnabled(false);
或者您可以更改定义 RecyclerView 的 xml 文件中的值:
<android.support.v7.widget.RecyclerView
android:id="@+id/yourRecyclerView"
android:layout_
android:layout_
android:nestedScrollingEnabled="false"/>
【讨论】:
感谢@Arman,我在 NestedScrollView 中使用了 RecyclerView,之后我得到了这个。但现在它工作正常。【参考方案4】:我刚遇到这个问题,我得到了
我将RecyclerView
放入ScrollView
所以请永远不要把RecyclerView
放在ScrollView
里面,这也可能导致。
mLayoutManager.findFirstVisibleItemPosition()
那总是返回一个固定值。并检查
recyclerview.setNestedScrollingEnabled(false);
【讨论】:
【参考方案5】:我通过覆盖calculateSpeedPerPixel(DisplayMetrics displayMetrics)
方法使滚动变得平滑:
public class CustomLayoutManager extends LinearLayoutManager
protected CenterLayoutManager(Context context, int orientation, boolean reverseLayout)
super(context, orientation, reverseLayout);
@Override
public void smoothScrollToPosition(RecyclerView recyclerView, RecyclerView.State state, int position)
RecyclerView.SmoothScroller smoothScroller = new CenterSmoothScroller(recyclerView.getContext());
smoothScroller.setTargetPosition(position);
startSmoothScroll(smoothScroller);
private class CenterSmoothScroller extends LinearSmoothScroller
CenterSmoothScroller(Context context)
super(context);
@Override
public int calculateDtToFit(int viewStart, int viewEnd, int boxStart, int boxEnd, int snapPreference)
return (boxStart + (boxEnd - boxStart) / 2) - (viewStart + (viewEnd - viewStart) / 2);
@Override
protected float calculateSpeedPerPixel(DisplayMetrics displayMetrics)
return 0.5f; //pass as per your requirement
【讨论】:
overwritecalculateSpeedPerPixel
为我工作,调用smoothScrollToPosition(position)
时默认会使recycler view 像跳转一样,但是当覆盖此方法并返回0.5f 时效果很好【参考方案6】:
android:animateLayoutChanges="false"
【讨论】:
那是什么?【参考方案7】:你需要使用“Picasso”库来异步加载图片,同时使用fit()
函数确保你加载的图片大小与布局中image view的大小相近。此外,如果您在滚动视图中使用回收器视图,请确保将 nested scrolling
设置为 false。
我使用以上三种方法解决了我的问题
【讨论】:
【参考方案8】:使用这里其他 cmets 所说的库。还要确保您没有嵌套在 Scrollview 中的 Recyclerviews。这也是我之前的不那么顺利的原因之一。
【讨论】:
【参考方案9】:用了 3 天,我解决了项目中的平滑滚动问题。
问题是<layer-list>
drawable 设置在item_user.xml
文件的背景中,因此需要 GPU 时间进行渲染,这就是滚动不流畅的原因。
不要在适配器项中使用<layer-list>
或任何其他复杂的可绘制对象。
【讨论】:
【参考方案10】:添加这个
animateLayoutChanges = true
在回收站视图 xml 中
【讨论】:
那是什么?以上是关于如何让 RecyclerView 平滑滚动?的主要内容,如果未能解决你的问题,请参考以下文章
平滑滚动不适用于 Android RecyclerView 的初始滚动
SwipeRefreshLayout 内的 NestedScrollView 内的 RecyclerView 不能平滑滚动