Оптимізація швидкості рендерингу UI мобільного додатку

TRUETECH займається розробкою, підтримкою та обслуговуванням мобільних додатків iOS, Android, PWA. Маємо великий досвід та експертизу для публікації мобільних додатків до популярних маркетів Google Play, App Store, Amazon, AppGallery та інші.

Розробка та підтримка будь-яких видів мобільних додатків:

Інформаційні та розважальні мобільні програми
Новинки, ігри, довідники, онлайн-каталоги, погодні, фітнес та здоров'я, туристичні, освітні, соціальні мережі та месенджери, квіз, блоги та подкасти, форуми, агрегатори
Мобільні програми електронної комерції
Інтернет-магазини, B2B-додатки, маркетплейси, онлайн-обмінники, кешбек-сервіси, біржі, дропшиппінг-платформи, програми лояльності, доставка їжі та товарів, платіжні системи
Мобільні програми для управління бізнес-процесами
CRM-системи, ERP-системи, управління проектами, інструменти для команди продажів, облік фінансів, управління виробництвом, логістика та доставка, управління персоналом, системи моніторингу даних
Мобільні програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, платформи надання електронних послуг, платформи кешбеку, відеохостинги, тематичні портали, платформи онлайн-бронювання та запису, платформи онлайн-торгівлі

Це лише деякі з типів мобільних додатків, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Послуги, які ми пропонуємо
Показано 1 з 1Усі 1735 послуг
Оптимізація швидкості рендерингу UI мобільного додатку
Складний
~3-5 днів
Часті запитання

Наші компетенції:

Етапи розробки

Останні роботи

  • image_mobile-applications_feedme_467_0.webp
    Розробка мобільного додатка для компанії FEEDME
    792
  • image_mobile-applications_xoomer_471_0.webp
    Розробка мобільного додатку для компанії XOOMER
    671
  • image_mobile-applications_rhl_428_0.webp
    Розробка мобільного додатку для компанії RHL
    1097
  • image_mobile-applications_zippy_411_0.webp
    Розробка мобільного додатку для компанії ZIPPY
    969
  • image_mobile-applications_affhome_429_0.webp
    Розробка мобільного додатку для компанії Affhome
    914
  • image_mobile-applications_flavors_409_0.webp
    Розробка мобільного додатку для компанії FLAVORS
    495

Оптимізація швидкості рендеринга UI мобільної програми

На iPhone 13 програма показувала 58–60 FPS на більшості екранів, але один екран з кастомним UICollectionViewLayout стабільно проседав до 42–45 FPS при скролу. Instruments показував, що 14 мс з 16 доступних уходило на layoutAttributesForElementsInRect — метод пересчитував всі позиції ячійок при кожному виклику без кешування. Це класика: UI-рендеринг не тормозит «взагалі», він тормозит в конкретному місці по конкретній причині.

Торможення рендеринга — одна з найнебезпечніших проблем, тому що вона видна користувачу негайно, але діагностується повільно. FPS-метрика говорить «погано», а де саме погано — потрібно розкапувати.

Де реально втрачаються кадри

Main thread — головний ворог плавності

Золоте правило — 16 мс на кадр (60 FPS) або 8 мс (120 Hz на Pro-пристроях). Все, що виконується на main thread понад це, блокує рендер. Типові винуватці:

На iOS: синхронна робота з CoreData через viewContext прямо в cellForItemAt, декодування UIImage без preparingForDisplay(), NSAttributedString з обчисленням розміру в sizeForItemAt без кешу.

На Android: блокуючий I/O в onBindViewHolder, Bitmap.decodeResource() на main thread, важкі Drawable анімації через AnimationDrawable на дешевих пристроях з Mali GPU.

Особняком стоїть проблема з measure/layout pass. На Android Jetpack Compose ConstraintLayout всередині LazyColumn з глибокою вкладеністю запускає два повні прохода measure на кожну ячійку. На складному списку з 50+ елементів це помітно навіть на Pixel 7.

GPU overdraw

Overdraw — це коли один пиксель малюється кілька разів за кадр. На Android включається через «Developer Options → Show GPU Overdraw»: синій — 1x, зелений — 2x, рожевий — 3x, червоний — 4x+. Червоний екран на бюджетному Xiaomi з Adreno 610 — гарантоване jank.

Часта причина — вкладені ViewGroup з непрозорими фонами, де кожен шар малює фон поверх попереднього. На iOS аналог — CALayer з opaque = false там, де прозорість не потрібна, або shouldRasterize без явного rasterizationScale.

Як ми діагностуємо та правимо

Робота починається з Xcode Instruments → Core Animation та Android GPU Inspector або вбудованого Android Studio Profiler → Rendering. Не з припущень — з даних.

Типовий сценарій на iOS-проекті: клієнт скаржиться на «торможення в ленті». Відкриваємо Time Profiler, записуємо скролл 5 секунд. У call tree одразу видно: [SDWebImage sd_setImageWithURL:] жре 8 мс на main thread тому що хтось убрав options:SDWebImageAvoidAutoSetImage та зображення застосовуються синхронно після завантаження. Один флаг — та FPS виросли з 47 до 59.

На Android був кейс з RecyclerView + DiffUtil: розробник викликав submitList() з ViewModel, але DiffUtil працював на main thread (використовувався ListAdapter без AsyncListDiffer). На списку з 200 елементів diff займав ~18 мс. Перенесли вичисленн diff на фоновий потік через AsyncListDiffer — проблема зникла.

Конкретні інструменти та техніки

iOS:

  • CADisplayLink + кастомний FPS-монітор у debug-сборці для постійного моніторингу
  • UIView.setNeedsLayout() vs UIView.layoutIfNeeded() — розуміння різниці критично при анімаціях
  • drawRect: майже завжди замінюємо на CALayer sublayers — Core Animation рендерить їх на GPU без учасити CPU
  • UIGraphicsImageRenderer замість застарілого UIGraphicsBeginImageContextWithOptions для offscreen rendering
  • Prefetching через UICollectionViewDataSourcePrefetching — декодуємо зображення до того, як ячійка з'явиться на екрані

Android / Compose:

  • Modifier.graphicsLayer {} для апаратного прискорення трансформацій замість програмного
  • remember {} та derivedStateOf {} — запобігають лишнім рекомпозиціям
  • key() у LazyColumn — без нього Compose не може переіспользовувати ноди при зміні списку
  • Bitmap.Config.RGB_565 замість ARGB_8888 там, де альфа-канал не потрібен — вдвічі менше пам'яті GPU

Flutter:

  • RepaintBoundary навколо віджетів, які часто перерисовуються незалежно
  • const конструктори — віджет не пересоздається при rebuild батька
  • flutter run --profile + DevTools → Performance overlay — обов'язковий інструмент перед релізом

Кейс: 120 Hz на iPad Pro

Клієнт зробив кастомну анімацію через UIViewPropertyAnimator з preferredFrameRateRange. Анімація працювала на 60 FPS замість 120. Виявилось — один CALayer з shouldRasterize = true без явного rasterizationScale = UIScreen.main.scale * 2. Core Animation ограничував весь subtree до 60 FPS через невідповідність масштабу растеризації. Після виправки анімація запрацювала на 120 FPS з помітною різницею в ощущеннях.

Етапи роботи

  1. Аудит — записуємо сесії в Instruments / Android Profiler, збираємо baseline-метрики FPS, janky frames, frame time
  2. Аналіз — виявляємо вузькі місця: main thread блокування, overdraw, лишні layout passes
  3. Правки — ітеративно, з замером після кожної зміни
  4. Регресійний прогон — перевіряємо, що правка не сломала сусідні екрани
  5. Моніторинг — інтегруємо Firebase Performance або власний FPS-монітор для відслідковування у продакшені

Оцінюємо об'єм після аудиту — іноді проблема розв'язується за день, іноді вимагає переписування кастомного layout.

Ориєнтири по строкам

Точкова правка (один екран, зрозуміла причина) — 1–3 дні. Системний аудит та оптимізація кількох екранів — 1–3 тижні. Якщо проблема в архітектурних рішеннях (неправильне використання main thread по всій програмі) — закладайте 3–6 тижнів з поетапною міграцією.