掌握 Android 中的 RecyclerView 优化

在本文中,我们将探讨各种技术和概念,帮助您充分利用 RecyclerView。

RecyclerView 池以及何时使用它

RecyclerView Pool 是一种帮助管理 RecyclerView 中视图的内存和性能的机制。它本质上是一个缓存,用于保存当前在屏幕上不可见但在不久的将来可能再次需要的视图。这显着减少了每次新项目进入可见区域时膨胀新视图的开销。

何时使用 RecyclerView 池:

  • 处理大型列表或数据网格时使用它。
  • 当列表中的项目具有不同的视图类型时,请选择 RecyclerView Pool。

onCreateViewHolder 和 onBindViewHolder

在RecyclerView中,onCreateViewHolderonBindViewHolder是适配器中必不可少的方法。它们共同为列表中的每个项目创建和绑定视图。

  • onCreateViewHolder在需要创建新视图时被调用。它负责扩展布局并创建 ViewHolder 实例。
  • 当现有视图被新项目重用时,将调用onBindViewHolder 。它将数据绑定到视图持有者,允许您更新视图的内容。
  • onBindViewHolder不应该用于绑定点击侦听器,我们应该注意我们在其中执行的匿名对象或任务。仔细阅读下面的代码,然后我会解释为什么?
class  MyAdapter ( private  val items: List<Item>) : RecyclerView.Adapter<MyAdapter.ViewHolder>() { 

    inside  class  ViewHolder (itemView: View) : RecyclerView.ViewHolder(itemView) { 
        init { 
            itemView.setOnClickListener { 
                // 处理项目click here 
                val position = adapterPosition 
                if (position != RecyclerView.NO_POSITION) { 
                    val clickedItem = items[position] 
                    // 处理 clickedItem 的点击事件
                } 
            } 
        } 
    } 

    override  fun  onCreateViewHolder (parent:ViewGroup , viewType: Int ) : ViewHolder {
        valview = LayoutInflater.from(parent.context).inflate(R.layout.item_layout,parent,false)
        returnViewHolder(view)
    }

    override  fun  onBindViewHolder (holder: ViewHolder ,position: Int ) {
        valcurrentItem = items[position]
        // 在此处将数据绑定到视图持有者
    }

    override  fun  getItemCount () = items.size
}
  • onBindViewHolder中,为每个项目单独生成单击侦听器,从而导致多个侦听器实例。通过将点击监听器放置在 中ViewHolder,它们在 ViewHolder 创建时就形成了,由于 ViewHolder 的可重用性,提高了内存效率。

优化 RecyclerView 的不同方法

  1. 使用图像库:处理图像时,建议使用提供位图池的图像库。这可以防止过多的内存使用和频繁的垃圾收集。
Glide.with( this ) 
    .load(imageUrl) 
    .placeholder(R.drawable.placeholder_image) 
    .error(R.drawable.error_icon) 
    .into(imageView)
  1. 优化图像尺寸:从服务器获取图像尺寸和宽高比,以避免不必要的调整大小和缩放。
  2. setHasStableIds:此方法应用于setHasStableIds(true)启用稳定的项目 ID。这有助于有效地更新和重新排序项目,而无需不必要的重新绑定。
class MyAdapter ( private val itemList: List<Item>) : RecyclerView.Adapter<MyAdapter.ViewHolder>() {

init {
setHasStableIds( true ) // 为此适配器启用稳定 ID
}

//....
}
  1. setHasFixedSize:如果RecyclerView大小本身是固定的并且不会因其内容而改变,则使用它setHasFixedSize(true)可以通过避免不必要的布局计算来帮助提高性能。参考:链接
val recyclerView = findViewById<RecyclerView>(R.id.recyclerView)
recyclerView.setHasFixedSize( true ) // 为 RecyclerView 启用固定大小
  1. setItemViewCacheSize:使用此方法调整缓存大小,以控制保留多少个离屏视图。这可以帮助管理内存使用。
val recyclerView = findViewById<RecyclerView>(R.id.recyclerView)
recyclerView.setItemViewCacheSize( 10 ) // 根据项目大小将视图缓存大小设置为自定义值

使视图无效与请求布局

  • Invalidate:用于指示视图的内容已更改并且需要重新绘制。
  • requestLayout:用于请求新的布局通道,影响视图的大小和位置。

您可能想知道为什么我突然谈论invalidaterequestLayout。让我解释。

我讨论了“无效”与“请求布局”,以说明它们如何影响视图的生命周期。这会影响屏幕重绘时间,过多的布局和绘制阶段会降低性能。要优化 Android 页面,尤其是 RecyclerView,彻底掌握视图生命周期至关重要。

让我用一个例子来进一步解释一下:

当 TextView 的 wrap_content 与 100dp 固定时,将或应该调用什么(invalidate 或 requestLayout)?

回答:

当 TextView 的宽度设置为 时wrap_content,调用requestLayout()会更合适。这是因为这requestLayout()表明视图层次结构需要重新测量和布局,以确保内容正确地适合边界。

当 TextView 的宽度固定时(例如,设置为特定值,如100dp),调用invalidate()通常就足够了。它通知系统视图的内容已更改,并且需要在现有布局范围内重新绘制。这可以避免在更新视图外观时进行不必要的重新布局计算。

ViewHolder模式

ViewHolder 模式是一种设计模式,它通过最大限度地减少对 .recyclerview 的调用次数来增强 recyclerview 的性能findViewById()。它涉及创建一个 ViewHolder 类,该类保存对项目布局中视图的引用。这允许在用户滚动列表时有效地重用视图。

有关 ViewHolder 模式的综合示例,请查看此链接

默认的废料和脏视图类型

RecyclerView中,每种视图类型的废视图和脏视图的默认计数设置为 5。对于标题等仅 1 或 2 个剪贴视图就足够的情况,此通用编号可能不是最佳选择。废料视图是可重用的、分离的视图,可以轻松地重新附加。

为了提高内存效率,您可以根据每个特定视图类型的流行程度自定义池大小,从而定制剪贴视图行为。这种方法通过有效管理RecyclerView中不同类型项目的剪贴视图数量,确保优化内存使用。val customRecycledViewPool = RecyclerView.RecycledViewPool().apply {
setMaxRecycledViews(viewType, poolSize) // 设置自定义 viewType 和 poolSize
}

结论

优化 RecyclerView 实施对于保持流畅且响应迅速的用户体验至关重要。通过理解 RecyclerView Pool 和 ViewHolder 模式等概念,并优化视图创建和绑定,您可以确保您的应用程序有效地处理大量数据。

请记住,每种优化技术都有特定的目的,并且应用这些技术的组合可以显着提高 RecyclerView 的性能。

阅读全文
下载说明:
1、本站所有资源均从互联网上收集整理而来,仅供学习交流之用,因此不包含技术服务请大家谅解!
2、本站不提供任何实质性的付费和支付资源,所有需要积分下载的资源均为网站运营赞助费用或者线下劳务费用!
3、本站所有资源仅用于学习及研究使用,您必须在下载后的24小时内删除所下载资源,切勿用于商业用途,否则由此引发的法律纠纷及连带责任本站和发布者概不承担!
4、本站站内提供的所有可下载资源,本站保证未做任何负面改动(不包含修复bug和完善功能等正面优化或二次开发),但本站不保证资源的准确性、安全性和完整性,用户下载后自行斟酌,我们以交流学习为目的,并不是所有的源码都100%无错或无bug!如有链接无法下载、失效或广告,请联系客服处理!
5、本站资源除标明原创外均来自网络整理,版权归原作者或本站特约原创作者所有,如侵犯到您的合法权益,请立即告知本站,本站将及时予与删除并致以最深的歉意!
6、如果您也有好的资源或教程,您可以投稿发布,成功分享后有站币奖励和额外收入!
7、如果您喜欢该资源,请支持官方正版资源,以得到更好的正版服务!
8、请您认真阅读上述内容,注册本站用户或下载本站资源即您同意上述内容!
原文链接:https://www.dandroid.cn/archives/21379,转载请注明出处。
0

评论0

显示验证码
没有账号?注册  忘记密码?