Осознанная оптимизация Compose 2: В борьбе с композицией

b904c5fcc6cc649a4b477a9743680c70.png

Jetpack Compose постоянно развивается, открывая перед разработчиками новые горизонты для оптимизации. С момента нашего последнего обзора, мы добились значительного прогресса, сократив задержки при скролле с 5–7% до нуля. В этом материале мы поделимся свежими находками и передовыми практиками в оптимизации Compose. Чтобы максимально углубиться в тему, рекомендуем ознакомиться с первой частью.

Серия статей:

  1. Осознанная оптимизация Compose

  2. Осознанная оптимизация Compose 2: В борьбе с композицией (текущая)

Содержание

Композиция — низвергнутый бог

Проблема начальной композиции

В первой части была описана одна из проблем ленивых списков — использование под капотом SubcomposeLayout. По словам разработчиков, ещё одной проблемой является скорость начальной композиции. Во время этого затратного этапа в первый раз строится дерево элементов. Для ленивых списков этот момент особенно критичен, поскольку начальная композиция происходит при создании каждого элемента.

В подкасте разработчики подчеркивают, что Compose проектировался исходя из того, что людям проще думать в категориях простых лейаутов (Box, Column, Row), а не таких сложных, как ConstraintLayout. В отличие от View, у Compose нет проблемы экспоненциального увеличения количества измерений, так как они ограничены одним проходом. Пропуски функций также способствуют решению проблемы вложенности. Но не в случае начальной композиции, когда нужно пройтись по тысяче лейаутов. Разработчики стремятся снизить и эту нагрузку: переход с версии Compose 1.4 на 1.6, по их словам, ускорит работу списков на 40%, что служит весомым аргументом в пользу обновления. Однако для уже оптимизированных списков прирост производительности может быть не так заметен.

Композиция в Compose обходится дорого, но это цена за удобство декларативного подхода. В данной статье мы предложим способы минимизации затрат на композицию и снижения их стоимости при рекомпозициях.

Для начала давайте посмотрим, как время композиции соотносится с другими фазами Compose:

Среднее время каждой фазы. Данные для слабого устройства. Источник

Среднее время каждой фазы. Данные для слабого устройства. Источник

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

Modifier.Node

Modifier.composed долгое время был основным инструментом для создания кастомных модификаторов. Однако, несмотря на его гибкость, этот подход имеет свои недостатки. Modifier.composed требует передачи composable-лямбды, внутри которой генерируется restartable-группа и множество других конструкций. Во время композиции под капотом вызываются эти composable-лямбды для получения конечного модификатора. Этот процесс называется материализацией, что усложняет начальную композицию. Кроме этого из-за лямбд страдало и сравнение модификаторов, так как каждый раз создаётся новая лямбда, которая не равна прошлой, следовательно цепочка модификаторов тоже считается другой из-за чего Compose делает меньше пропусков.

Теперь на помощь приходит Modifier.Node — новый, более эффективный способ создания кастомных модификаторов. Этот подход позволяет реализовать все задачи, для которых ранее использовался Modifier.composed, но без лишних накладных расходов, связанных с composable-лямбдами и композицией. Самое важное преимущество — классы, созданные с помощью Modifier.Node, могут быть сравнены и переиспользованы, что значительно улучшает производительность.

Google недавно обновила документацию, добавив множество примеров и рекомендаций по использованию Modifier.Node. Это отличный повод пересмотреть существующие реализации модификаторов в вашем проекте и заменить их на более оптимизированные Modifier.Node.

Проблемы DerivedState и remember

Как мы уже выяснили, композиция сама по себе дорогая и стоит использовать DerivedState, чтобы ограничить число рекомпозиций. Однако часто разработчики применяют его некорректно, что было подробно разобрано в первой части. Сейчас стоит объяснить, почему так важно следить за этим. 

Рассмотрим время, затрачиваемое на чтение тех или иных данных, чтобы понять масштабы проблемы (источник):

  • Чтение локальной переменной — 1 нс

  • Чтение из поля класса — 5 нс

  • Вызов метода — 10 нс

  • Чтение с синхронизацией — 50 нс

  • map.get — 150 нс

  • state.value — 2 500 нс

  • derivedState.value — 10 000 нс

Учитывая, что для поддержания частоты обновления экрана в 120 кадров в секунду максимальное время отрисовки одного кадра не должно превышать 8 333 333 нс, становится очевидным, что неправильное использование DerivedState может значительно замедлить ваше приложение. Если такой подход не приводит к заметному уменьшению числа рекомпозиций, это может сделать код медленнее, чем при прямом обращении к исходному State.

Та же логика применима и к функции remember { }. Необдуманное «запоминание» простых вычислений может неоправданно замедлить работу приложения. Например, чтобы запомнить обычное выражение, потребуется сперва сравнить ключи с их предыдущими значениями, что уже само по себе будет дороже, и это не говоря про чтение и запись в Slot Table.

// Антипример
val expr = remember(a, b, c) { a + b * c }

Поэтому и нужно хоть немного знать, как Compose работает под капотом, чтобы не использовать его функции себе во вред.

Пересядь с иглы Compose на старый добрый Kotlin

Для эффективной работы с Jetpack Compose ключевым является умение находить золотую середину между использованием возможностей Compose и стандартным Kotlin кодом. Как отметил разработчик Compose Андрей Шиков: «Изучая Compose, мы позабыли как использовать Kotlin». Давайте взглянем на его пример, демонстрирующий этот подход. В исходной версии кода использовалось множество отдельных состояний и сайд-эффектов:

@Composable 
fun MyComponent(modifier: Modifier, config: Config) {
    val interactionSource = remember { MutableInteractionSource() }
    val activeInteractions = remember { mutableStateListOf() }
    val config by rememberUpdatedState(config)

    LaunchedEffect(interaction Source) {
        interactionSource.interactions.collect {
            // Обновление списке взаимодействий
        }
    }

    val animatableColor = remember {
        Animatable(config.defaultColor, Color.VectorConverter)
    }

    LaunchedEffect(config) {
        // Обновление animatableColor
    }

    LaunchedEffect(interactions) {
        snapshotFlow { activeInteractions.lastOrNull() }.collect {
            // Обновление animatableColor на основе взаимодействий и конфига
        }
    }
    
    // Использование animatableColor при отрисовке
}

Этот код был оптимизирован путём объединения нескольких состояний и сокращения количества сайд-эффектов за счёт переноса логики в одну корутину:

@Composable 
fun MyComponent(modifier: Modifier, config: Config) {
    val state = remember { MyComponentState(config) }

    LaunchedEffect(state) {
        state.collectUpdates()
    }

    SideEffect {
        state.config.value = config
    }

    // Использование state.animatableColor при отрисовке
}

Благодаря такому рефакторингу удалось ускорить работу кода на 9%.

Пререндер

При первом открытии экрана приложения, время ожидания пользователя состоит из двух основных этапов: загрузки данных и отрисовки UI. Для минимизации времени ожидания появления контента на экране можно использовать технику пререндера.

Суть метода заключается в том, что во время загрузки данных параллельно происходит отрисовка экрана с использованием моковых данных. Очень важно, чтобы структура композиции UI не менялась сильно после замены моковых данных на реальные. Учитывая высокую стоимость процесса композиции, такой подход позволяет заранее «прогреть» композицию, существенно сокращая время до появления актуального контента на экране.

Один из вариантов реализации пререндеринга — использование индикатора загрузки, отрисованного поверх всего экрана:

ProductDetailScreen(state: ProductUiState) {
    // Вначале state содержит пустые данные для пререндера контента
    ProductDetailContent(state)

    // Отображение индикатора загрузки поверх контента, а не вместо него
    if (state.isLoading) {
        FullscreenLoader()
    }
}

Также можно применять эффект шиммера на моковых данных, используя модификатор placeholder из библиотеки accompanist. Этот метод не влияет на структуру композиции после загрузки реальных данных и обеспечивает плавный визуальный переход. Однако, он требует дополнительной адаптации существующих элементов для корректной отрисовки шиммера поверх контента.

В контексте использования ленивых списков особое внимание следует уделить однородности элементов с одинаковым contentType. Если элементы списка не будут сильно отличаться по структуре композиции, это снизит необходимость в дополнительной работе по перестройке дерева композиции при их переиспользовании, что также способствует ускорению скролла.

Отложенная композиция

Ленивые лейауты используют SubcomposeLayout для определения того, какие элементы должны быть отображены на экране. Для этого происходит внутренняя композиция во время фазы компоновки (layout). Этот процесс происходит за один кадр, что может стать проблемой при работе с тяжёлыми экранами. 

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

В примере ниже демонстрируется как отложить композицию части экрана, используя метод withFrameNanos { }, который аналогично delay() останавливает корутины, но делает это ровно до начала следующего кадра:

@Composable
fun ProductDetail(
    productInfo: ProductInfo
) {
    var blockState by remember { mutableStateOf(0) }

    LaunchedEffect(Unit) {
        while (blockState < 3) {
            // Откладываем композицию каждого блока на 1 кадр
            withFrameNanos {  }
            blockState += 1
        }
    }

    ProductBlock0(productInfo)

    if (blockState >= 1) {
        ProductBlock1(productInfo)
    }
    if (blockState >= 2) {
        ProductBlock2(productInfo)
    }
    if (blockState >= 3) {
        ProductBlock2(productInfo)
    }
}

Painter

Xml-иконки vs Compose-иконки

При работе с иконками в Compose существует два основных подхода: использование иконок в формате XML и прямое создание иконок с помощью кода. Рассмотрим процесс и эффективность каждого из них.

// Xml-иконка
Image(
    painter = painterResource(R.drawable.my_icon), 
    contentDescription = null
)

// Compose-иконка
Image(
    painter = rememberVectorPainter(Icons.Filled.Home), 
    contentDescription = null
)

Для XML иконки процесс загрузки включает вызов painterResource(R.drawable.my_icon), состоящий из следующих шагов:

  1. Чтение ресурса из файла или получение его из кэша.

  2. Преобразования XML в ImageVector.

  3. Создание Painter с помощью функции rememberVectorPainter().

В случае с иконками, созданными непосредственно в Compose, процесс выглядит следующим образом:

  1. Вызов Icons.Filled.Home сразу инициирует создание объекта ImageVector.

  2. Создание Painter с помощью функции rememberVectorPainter().

Тесты показали, что иконки, созданные с помощью Compose, загружаются от 5% до 18% быстрее по сравнению с иконками в формате XML. Скорость загрузки напрямую зависит от сложности структуры иконки и размера исходного файла. Важно отметить, что общее влияние иконок на производительность приложения может сильно варьироваться в зависимости от их количества на экране и использования в ленивых списках.

Для создания собственных Compose-иконок можно воспользоваться инструментом SVG to Compose, который поддерживает конвертацию как SVG, так и XML файлов. Также стоит упомянуть, что эти иконки убавят вам проблем с ресурсами при использовании Compose Multiplatform.

Нестабильность Painter

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

Вынос Painter

Вынесение создания объекта Painter за пределы списка позволяет заметно увеличить скорость отрисовки. Это особенно критично в списках, где каждый элемент требует инициализации собственного Painter. Несмотря на наличие кэша для XML ресурсов, каждый такой вызов создает дополнительную нагрузку. Создание единого Painter для всего списка значительно уменьшает эту нагрузку. Однако, как и в случае с выносом модификаторов, может  пострадать читабельность кода.

Пример до оптимизации:

@Composable
fun MyList(products: ImmutableList) {
    LazyColumn { 
        items(products) { product -> 
            // Painter внутри элемента списка
            MyProductItem(product, painterResource(R.drawable.ic_menu)) 
        } 
    }
}

Пример после оптимизации:

@Composable
fun MyList(products: ImmutableList) {
    // Выносим Painter для общей иконки из списка
    val menuPainter = painterResource(R.drawable.ic_menu)

    LazyColumn { 
        items(products) { product -> 
            MyProductItem(product, menuPainter) 
        } 
    }
}

Этот метод эффективен не только для иконок, но и для других ресурсов, хотя для последних прирост производительности может быть менее заметен. Подход актуален не только для списков, но и для мест с частой рекомпозицией из-за анимаций, где рекомендуется вообще избегать создания дорогих объектов.

Дизайн система

Создание дизайн-системы является ключевым этапом при внедрении Jetpack Compose в проекты. Разработчики на этом этапе могут столкнуться с проблемами из-за недостаточного опыта разработки на новой технологии, что ведет к появлению неэффективного кода в важнейших частях дизайн-системы.

Цветовая схема

Традиционно при создании кастомной цветовой палитры разработчики копируют подход, используемый в MaterialTheme. Однако недавние коммиты разработчиков указывают на недостатки такой реализации.

В традиционной реализации для каждого цвета создавался отдельный State, что вело к необходимости подписки на изменение состояния каждый раз при его чтении. Это могло негативно повлиять на производительность, особенно если в дизайн-системе использовалось множество цветов.

Пример до оптимизации:

@Stable
class ColorScheme(
    primary: Color,
    onPrimary: Color,
) {
    // State для каждого цвета
    var primary by mutableStateOf(primary)
        internal set
    var onPrimary by mutableStateOf(onPrimary)
        internal set
}

Изначально такая реализация имела преимущество в гибкости — возможность изменять каждый цвет отдельно без значительных затрат на производительность. Однако, как показывает практика, в большинстве приложений цветовая схема меняется лишь при переключении между светлой и темной темами, а не при индивидуальном изменении отдельных цветов.

Переход к использованию обычного data class для описания цветовой схемы устраняет необходимость в подписках на состояние и улучшает производительность приложения.

Пример после оптимизации:

@Immutable
data class ColorScheme(
    val primary: Color,
    val onPrimary: Color,
)

Ошибки прошлого

Код, о котором будет идти речь, был написан давно и с тех пор не подвергался изменениям. Однако именно он использовался на всех наших экранах и существенно влиял на время отрисовки кадров из-за неэффективной работы с типографией в AppTheme.

AppTheme.typography создаёт новую типографию при каждом вызове, загружая шрифты и цвета из ресурсов для каждого стиля текста отдельно. Это приводило к множественным обращениям к ресурсам и чтению AppTheme.colors.textPrimary 16 раз для каждого TextStyle.

object AppTheme {
    @Composable
    @ReadonlyComposable
    val typography
        get() = DefaultAppTypography
}

@Composable
@ReadonlyComposable
val DefaultAppTypography
    get() = AppTypography(
        headXXL = TextStyle(
            color = AppTheme.colors.textPrimary,
            ...
        ),
        // ...
        // Создание 15 других стилей текста
    )

Эта реализация особенно заметно замедляла отрисовку на экранах с большим количеством текста и различными стилями, поскольку каждый текстовый элемент инициировал вызов условного AppTheme.typography.headXXL. Нашли мы эту проблему после исследования нескольких экранов с помощью трассировки композиции.

Решение проблемы заключалась в изменении подхода к созданию и использованию типографии. Теперь типография AppTheme инициализируется единожды и доступна через LocalAppTypography.current, что значительно сокращает количество обращений к ресурсам и ускоряет работу с типографией во всем приложении:

object AppTheme {
    @Composable
    @ReadonlyComposable
    val typography
        get() = LocalAppTypography.current
}

Форматтеры

Важно помнить о правильном размещении логики форматирования чисел и валют. Интуитивно может показаться, что использование форматирования непосредственно в коде Compose — это удобно. Однако, этот подход может привести к неожиданным проблемам производительности. Рекомендуется переносить создание и использование форматтеров в бизнес-логику вашего приложения. Такой шаг позволяет не только уменьшить нагрузку на главный поток, но и избежать излишнего дублирования объектов форматтера, обеспечивая их переиспользование между экранами. Этот совет кажется простым, но на практике часто упускается из виду, особенно когда форматирование спрятано за утилитарные функции. 

Антипример форматирования в Compose-коде:

Text(
    text = productItem.price.toMoneyFormat()
)

Однако, необходимо учитывать, что оптимизация за счёт переноса форматирования может не всегда давать ожидаемый результат. В случае обработки списка в бизнес-логике, форматирование применяется ко всему списку сразу, в то время как в Compose, в контексте ленивых списков, форматирование выполняется исключительно для элементов, видимых пользователю. Это означает, что перемещение логики форматирования может потенциально увеличить время до отображения содержимого на экране. Важно помнить, что не каждая оптимизация ведёт к улучшению производительности, и нужно тщательно анализировать изменения, прежде чем их внедрять

Оптимизация на спичках

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

Автоупаковка

Автоупаковка примитивных типов данных может незаметно замедлить выполнение кода, особенно когда он используется в большом количестве мест. В Jetpack Compose разработчики активно стремятся минимизировать подобные затраты, применяя различные способы:

  • Использование специальных MutableState для примитивных типов (например, mutableIntStateOf()) помогает избежать упаковки.

  • Введение специальных значений (Unspecified) для определенных классов вместо null позволяет избежать автоупаковки благодаря инлайнингу value классов. Например, недавно добавили Unspecified для TextAlign, TextDirection и др., чтобы избежать null.

  • Замена Pair на value class с полем типа Long значительно снижает затраты на хранение данных, используя стек вместо кучи. Тип Long содержит в два раза больше бит, чем Int, что позволяет ему хранить два числа типа Int и обращаться к ним с помощью побитовых операций.

Быстрые методы

Не все стандартные методы Kotlin идеально подходят для конкретных задач. Jetpack Compose предлагает альтернативы, например, fastForEach или fastFirst. О быстрых методах можно прочитать в блоге Romain Guy.

Эффективные структуры

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

  • Например, mutableListOf использует ArrayList, что может быть избыточным, если не требуется динамическое изменение размера коллекции или использование дженериков. В критически важных частях кода лучше применять специализированные массивы (например, IntArray). 

  • Метод mutableMapOf по умолчанию создаёт LinkedHashMap, что может быть менее эффективным, чем другие типы данных. Jetpack Compose использует, например, ScatterMap

Внутри AndroidX Collections много оптимизированных структур, которые стоит использовать в критическом коде.

Вложенные if

При работе с условными конструкциями важно учитывать влияние вложенности на производительность. Каждая условная ветка создаёт дополнительные вызовы replaceable-группы для поддержки быстрой замены кода. Поэтому в случаях, когда вложенность условных блоков не является необходимостью, предпочтение следует отдавать плоской структуре. Это позволяет не только упростить код, но и повысить его производительность за счёт уменьшения количества операций.

Пример генерации кода для плоских if (также и для when):

if (condition1) {
    $composer.startReplaceableGroup()
    Content1()
    $composer.endReplaceableGroup()
} else if (condition2) {
    $composer.startReplaceableGroup()
    Content2()
    $composer.endReplaceableGroup()
} else {
    $composer.startReplaceableGroup()
    Content3()
    $composer.endReplaceableGroup()
}

Пример генерации кода для вложенных if:

if (condition1) {
    $composer.startReplaceableGroup()
    Content1()
    $composer.endReplaceableGroup()
} else {
    $composer.startReplaceableGroup()
    if (condition2) {
        $composer.startReplaceableGroup()
        Content2()
        $composer.endReplaceableGroup()
    } else {
        $composer.startReplaceableGroup()
        Content3()
        $composer.endReplaceableGroup()
    }
    $composer.endReplaceableGroup()
}

Нововведения

Указание стабильности внешних типов

С релизом Compose Compiler версии 1.5.5 появилась возможность явно указывать стабильность внешних типов. Это нововведение позволяет избежать использования дополнительных обёрток для обеспечения стабильности. Рекомендуем добавить в список стабильных типов такие часто используемые классы, как стандартные коллекции из Kotlin и Painter. Это нужно указать в каждом модуле, где используется Compose. Указание коллекций особенно актуально для тех, кто предпочитает не использовать immutable коллекции из-за нахождения их в альфа-версии или из-за необходимости переписывания большого объема кода.

// Все коллекции из котлин 
kotlin.collections.* 

// Painter 
androidx.compose.ui.graphics.painter.Painter

Режим сильной стабильности

На данный момент режим сильной стабильности является экспериментальным и активируется через специальный флаг. В перспективе он может стать настройкой по умолчанию. Что он даёт:

  • Все перезапускаемые (restartable) функции станут пропускаемыми (skippable). Для нестабильных параметров сравнение по экземплярам, для стабильных — через equals.

  • Лямбды, захватывающие нестабильные переменные, будут тоже обёрнуты в remember.

Инструментарий

Просмотр исходного кода Compose

Каждый раз, когда вы сомневаетесь в том, как будет работать тот или иной код после компилятора Compose, стоит просто посмотреть финальный Java код. Хоть вы и можете отлично знать генерацию Compose-кода из Jetpack Compose Internals или множества статей, это не спасёт вас от устаревания информации там. Для этого есть удобный gradle-плагин — decomposer.

Vkcompose плагин

Среди полезных инструментов есть плагины от VK для Kotlin и IDE, которые выполняют:

  • Подсветку нестабильных параметров и непропускаемых функций непосредственно в IDE.

  • Визуальное выделение происходящих рекомпозиций в UI с помощью цветных границ.

  • Логирование причин, по которым произошла рекомпозиция.

Detekt

Detekt, позволяет не только следить за стилем кода, но и защищать от не очень хороших практик, в том числе приводящих к проседанию производительности.

  • compose-rules — большой список разнообразных правил.

  • vkcompose — кроме плагина предоставляет правило, которое проверяет функции на пропускаемость.

Итог

Подводя итог, мы разобрались, как избегать проблемы с начальной композицией и что не стоит перегружать Compose лишней логикой, ведь за удобство и простоту приходится платить. Однако благодаря неустанной работе разработчиков, производительность Compose значительно выросла, давая нам свободу сосредоточиться на других аспектах разработки. А экраны с огромным DAU и на View приходилось оптимизировать за гранью обычных приёмов. Для Compose это далеко не предел: будущие оптимизации сделают его ещё более мощным инструментом, идеально подходящим для любых задач.

© Habrahabr.ru