Recyclerview может быть настолько гладким, что нельзя пропустить эти 14 стратегий оптимизации...
Recyclerview может быть настолько гладким, что нельзя пропустить эти 14 стратегий оптимизации...

введение

В разработке для Android RecyclerView — это часто используемый элемент управления списком, используемый для отображения больших объемов данных. Однако по мере увеличения объема данных производительность RecyclerView может снизиться, что приведет к таким проблемам, как задержки и утечки памяти. В этой статье будут представлены некоторые методы оптимизации, которые помогут вам повысить производительность RecyclerView, чтобы она оставалась бесперебойной при различных обстоятельствах.

Идеи по оптимизации

Основные идеи оптимизации производительности RecyclerView можно свести к следующим аспектам:

  1. Оптимизация макета: оптимизация RecyclerView измакетструктура,Уменьшить уровень вложенности,Повышение эффективности макета.
  2. Уменьшить рисунок: максимально уменьшите количество прорисовок вида, чтобы избежать потребления изпроизводительности, вызванного перерисовкой.
  3. Скользящая оптимизация: существовать Во время скольжения,Максимально сократить трудоемкие операции,Избегайте влияния на эффект скольжения.
  4. предварительная загрузка: предварительная загрузка скоро будет из просмотра,Улучшение отображенияпроизводительность。
  5. Оптимизация памяти: Уменьшите потребление памяти.,Разумно освободить память,Избегайте утечек памяти.

Ниже приведены конкретные стратегии оптимизации для каждого из них.

Оптимизация макета

  1. Уменьшите вложенность макетов

Избегайте использования слишком большого количества вложенных макетов и сложных иерархий в макете элемента RecyclerView, что увеличит время и потребление ресурсов рендеринга. Попробуйте использовать простую структуру макета и рационально использовать эффективные макеты, такие как ConstraintLayout.

Язык кода:javascript
копировать
<!-- item_layout.xml -->
<androidx.constraintlayout.widget.ConstraintLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"/>

    <!-- Другие компоненты представления -->

</androidx.constraintlayout.widget.ConstraintLayout>
  1. Используйте тег слияния для объединения макета

Используйте тег слияния, чтобы объединить несколько файлов макета в один, уменьшив уровни макета и повысив производительность рисования.

Язык кода:javascript
копировать
<!-- Используйте тег слияния для объединения макета -->
<merge xmlns:android="http://schemas.android.com/apk/res/android">
    <ImageView
        android:id="@+id/imageView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/image" />

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Text" />
</merge>
  1. Включить setHasFixedSize

настраивать setHasFixedSize(true) назад,RecyclerView предполагает, что высота всех элементов фиксирована.,Перерасчет не произойдет из-за изменений в Itemiz,избегатьrequestLayoutпривести киз Пустая трата ресурсов。

Язык кода:javascript
копировать
val recyclerView = findViewById<RecyclerView>(R.id.recyclerView)
recyclerView.setHasFixedSize(true)

Следует отметить, что использование setHasFixedSize(true)Применяется ко всемItemВысота фиксирована и не меняется.из Состояние。еслиItemВысота не фиксирована или может меняться.,Этот метод следует избегать,В противном случае макет может отображаться ненормально.

Уменьшить рисунок

  1. Использование DiffUtil для обновления данных

Когда набор данных изменяется, использование DiffUtil для расчета разницы может уменьшить количество ненужных обновлений пользовательского интерфейса и повысить производительность. DiffUtil может эффективно вычислять различия наборов данных в фоновом потоке и применять результаты к RecyclerView.

Язык кода:javascript
копировать
class MyDiffCallback(private val oldList: List<String>, private val newList: List<String>) : DiffUtil.Callback() {
    override fun getOldListSize(): Int {
        return oldList.size
    }

    override fun getNewListSize(): Int {
        return newList.size
    }

    override fun areItemsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
        return oldList[oldItemPosition] == newList[newItemPosition]
    }

    override fun areContentsTheSame(oldItemPosition: Int,newItemPosition: Int): Boolean {
        return oldList[oldItemPosition] == newList[newItemPosition]
    }
}

// существоватьAdapterПрименение вDiffUtil
val diffResult = DiffUtil.calculateDiff(MyDiffCallback(oldList, newList))
diffResult.dispatchUpdatesTo(this)
  1. Ограничьте количество элементов списка

Если объем данных в списке очень велик, вы можете рассмотреть возможность загрузки страниц или загрузки данных только в пределах видимого диапазона, чтобы уменьшить использование памяти и время рендеринга.

Язык кода:javascript
копировать
// Загружать только видимый диапазон изданных
recyclerView.layoutManager?.setInitialPrefetchItemCount(10)

Скользящая оптимизация

  1. существоватьonCreateViewHolderсередина Выполните необходимые операции инициализации

существуютViewHolderиз Этап создания,Выполните необходимые операции инициализации,Такие как настройки монитора и т.д.,избегатьсуществоватьonBindViewHolder()середина进行耗时操作,Улучшение прокруткипроизводительность。

Язык кода:javascript
копировать
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
    val view = LayoutInflater.from(parent.context).inflate(R.layout.item_layout, parent, false)
    val viewHolder = ViewHolder(view)
    // Выполните необходимые операции инициализации
    return viewHolder
}
  1. Сдвиньте, чтобы остановить операцию загрузки

может пройти RecyclerView.addOnScrollListener(listener) Метод добавляет прослушиватель прокрутки, а затем выполняет в нем соответствующие операции для дальнейшей оптимизации эффекта скольжения.

Язык кода:javascript
копировать
val recyclerView = findViewById<RecyclerView>(R.id.recyclerView)

val layoutManager = LinearLayoutManager(this)
recyclerView.layoutManager = layoutManager

val adapter = MyAdapter(dataList)
recyclerView.adapter = adapter

recyclerView.addOnScrollListener(object : RecyclerView.OnScrollListener() {
    override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
        super.onScrollStateChanged(recyclerView, newState)
        // Определите, является ли состояние прокрутки состоянием остановки прокрутки.
        if (newState == RecyclerView.SCROLL_STATE_IDLE) {
            startLoaidng()
        } else {
            // Выполните операции остановки загрузки, например остановку загрузки изображения и т. д.
            stopLoading()
        }
    }
})

предварительная загрузка

  1. Начать расчетExtraLayoutSpace

calculateExtraLayoutSpace Этот метод можно использовать для увеличения дополнительного пространства, зарезервированного RecyclerView, что помогает заранее загружать элементы за пределами экрана и избегать лагов во время скольжения.

тыможет пройтипереписатьcalculateExtraLayoutSpaceметод возврата дополнительныхиз Размер пространства,так чтоRecyclerViewсуществовать Во время скольженияпредварительная загрузка закадрового из Предмета.

Язык кода:javascript
копировать
class CustomLayoutManager : LinearLayoutManager {

    constructor(context: Context) : super(context)

    constructor(context: Context, orientation: Int, reverseLayout: Boolean) : super(context, orientation, reverseLayout)

    override fun calculateExtraLayoutSpace(state: RecyclerView.State, extraLayoutSpace: IntArray) {
        super.calculateExtraLayoutSpace(state, extraLayoutSpace)
        // обустройство дополнительного пространства измакета, которое можно динамически рассчитывать по мере необходимости
        extraLayoutSpace[0] = 200 
        extraLayoutSpace[1] = 200
    }
}
  1. Переопределить CollectAdjacentPrefetchPositions

collectAdjacentPrefetchPositionsМетодRecyclerViewсерединаизметод защиты,Используется для сбора предварительно выбранных местоположений, соседних с заданным местоположением. Этот метод в основном используется для механизма предварительной выборки RecyclerView.,Используется для существования предварительной выборки рядом с текущей позицией во время скольжения из данных.,Улучшите плавность скольжения.

ты можешьсуществовать НастроитьLayoutManagerсерединапереписатьcollectAdjacentPrefetchPositionsМетод достижения соседнего положенияизлогика предварительной выборки。

Язык кода:javascript
копировать
class CustomLayoutManager : LinearLayoutManager {

    constructor(context: Context) : super(context)

    constructor(context: Context, orientation: Int, reverseLayout: Boolean) : super(context, orientation, reverseLayout)

    override fun collectAdjacentPrefetchPositions(dx: Int, dy: Int, state: RecyclerView.State?, layoutPrefetchRegistry: LayoutPrefetchRegistry) {
        super.collectAdjacentPrefetchPositions(dx, dy, state, layoutPrefetchRegistry)

        // По направлению скольжения (dx, dy) Собрать соседние позиции предварительной выборки
        val anchorPos = findFirstVisibleItemPosition()
        if (dy > 0) {
            // Проведите пальцем вниз, чтобы предварительно загрузить следующие объекты:
            for (i in anchorPos + 1 until state?.itemCount ?: 0) {
                layoutPrefetchRegistry.addPosition(i, 0)
            }
        } else {
            // Проведите пальцем вверх, чтобы предварительно загрузить указанные выше элементы.
            for (i in anchorPos - 1 downTo 0) {
                layoutPrefetchRegistry.addPosition(i, 0)
            }
        }
    }
}

Оптимизация памяти

  1. Общий RecyclerViewPool

Если несколько RecycledView из Adapter Как и в случае с RecycledViewPool, вы можете использовать общий доступ к RecycledViewPool между RecyclerViews, чтобы повысить производительность.

Язык кода:javascript
копировать
// Создайте общий изRecycledViewPool.
val recycledViewPool = RecyclerView.RecycledViewPool()

// настройки Поделиться изRecycledViewPool на несколько RecyclerView
recyclerView1.setRecycledViewPool(recycledViewPool)
recyclerView2.setRecycledViewPool(recycledViewPool)

Этот подход особенно подходит для ситуаций, когда структура нескольких RecyclerViewPool очень схожа. Совместное использование RecycledViewPool может еще больше повысить производительность RecyclerView.

  1. Используйте Adaptor.setHasStableIds(true) для улучшения стабильности элемента.

настраиватьAdapterизsetHasStableIds(true)можно улучшитьItemизстабильность,Помогите RecyclerView лучше идентифицировать и повторно использовать ViewHolder.,Избегайте частого создания и удаления ViewHolder.,Уменьшите потребление памяти.

Язык кода:javascript
копировать
adapter.setHasStableIds(true)
  1. Используйте RecyclerView.setItemViewCacheSize(size) для настройки размера кэша

проходитьнастраиватьRecyclerViewизsetItemViewCacheSize(size)методнастраиватьразмер кэша,Может контролировать количество кэшированных ViewHolder в RecyclerView.,Избегайте слишком большого количества кэшей, занимающих слишком много памяти.

Язык кода:javascript
копировать
recyclerView.setItemViewCacheSize(20) // размер кэша настроек равен 20
  1. Общие события

Например событие клика,Можно создать общий объект прослушивателя,И дать ему настройки всем изItemView. Затем различайте и выполняйте различные операции на основе идентификатора. Это позволяет избежать создания объекта прослушивателя для каждого элемента.,оптимизация снизила потребление ресурсов.

Язык кода:javascript
копировать
// Общий объект прослушивателя
val itemClickListener = View.OnClickListener { view ->
    // Выполнение различных операций на основе идентификатора представления
    when (view.id) {
        R.id.button -> {
            // Выполнить действие по нажатию кнопки
        }
        R.id.imageView -> {
            // Выполнение операций щелчка изображения
        }
        // Другая обработка IDиз...
    }
}

// существованиеViewHolder для ItemViewнастройки, общие из прослушивателя
inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
    init {
        // Общий доступ для всех тех, кому нужна настройка ItemView
        itemView.setOnClickListener(itemClickListener)
    }
}
  1. Переопределить RecyclerView.onViewRecycled(holder) для переработки ресурсов

существовать onViewRecycled(holder: ViewHolder) 方法середина,Мы можем выполнить некоторые операции по освобождению ресурсов.,Например Освободите ресурсы изображения в ViewHolder、Удалить прослушиватели и т. д.,Чтобы связанные ресурсы могли быть освобождены вовремя, когда существующийViewHolder будет переработан.,Избегайте утечек памяти и непроизводительной траты ресурсов.

Язык кода:javascript
копировать
override fun onViewRecycled(holder: ViewHolder) {
    super.onViewRecycled(holder)
    // Освободите ресурсы изображения в ViewHolder
    holder.imageView.setImageDrawable(null)
    // Удалить прослушиватель в ViewHolder
    holder.itemView.setOnClickListener(null)
}

Подвести итог

Выбрав правильный изоптимизациямакет, Уменьшить рисунок、Скользящая оптимизация、предварительная загрузкаи Оптимизация памяти Стратегия,Может эффективно улучшить производительность RecyclerView.,Сделайте его существование гладким в любых ситуациях. существуютфактически в разработке,Также необходимо выбрать подходящую стратегию изоптимизации в соответствии с конкретной ситуацией.,и выполнить соответствующие испытания и настройки,Добиться наилучшего эффекта изпроизводительности.

boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose