Нативная разработка Android приложений

TRUETECH занимается разработкой, поддержкой и обслуживанием мобильных приложений iOS, Android, PWA. Имеем большой опыт и экспертизу для публикации мобильных приложений в популярные маркеты Google Play, App Store, Amazon, AppGallery и другие.
Разработка и поддержка любых видов мобильных приложений:
Информационные и развлекательные мобильные приложения
Новостные приложения, игры, справочники, онлайн-каталоги, погодные, фитнес и здоровье, туристические, образовательные, социальные сети и мессенджеры, квиз, блоги и подкасты, форумы, агрегаторы
Мобильные приложения электронной коммерции
Интернет-магазины, B2B-приложения, маркетплейсы, онлайн-обменники, кэшбэк-сервисы, биржи, дропшиппинг-платформы, программы лояльности, доставка еды и товаров, платежные системы
Мобильные приложения для управления бизнес-процессами
CRM-системы, ERP-системы, управление проектами, инструменты для команды продаж, учет финансов, управление производством, логистика и доставка, управление персоналом, системы мониторинга данных
Мобильные приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, платформы предоставления электронных услуг, платформы кешбека, видеохостинги, тематические порталы, платформы онлайн-бронирования и записи, платформы онлайн-торговли

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

Предлагаемые услуги
Показано 30 из 32 услугВсе 1735 услуг
Простая
от 4 часов до 2 рабочих дней
Сложная
от 2 недель до 3 месяцев
Сложная
от 2 недель до 3 месяцев
Сложная
от 2 недель до 3 месяцев
Простая
от 1 рабочего дня до 3 рабочих дней
Средняя
от 1 рабочего дня до 3 рабочих дней
Средняя
от 1 рабочего дня до 3 рабочих дней
Средняя
от 1 рабочего дня до 3 рабочих дней
Средняя
от 1 рабочего дня до 3 рабочих дней
Средняя
~3-5 рабочих дней
Простая
от 1 рабочего дня до 3 рабочих дней
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_mobile-applications_feedme_467_0.webp
    Разработка мобильного приложения для компании FEEDME
    756
  • image_mobile-applications_xoomer_471_0.webp
    Разработка мобильного приложения для компании XOOMER
    624
  • image_mobile-applications_rhl_428_0.webp
    Разработка мобильного приложения для компании RHL
    1052
  • image_mobile-applications_zippy_411_0.webp
    Разработка мобильного приложения для компании ZIPPY
    947
  • image_mobile-applications_affhome_429_0.webp
    Разработка мобильного приложения для компании Affhome
    862
  • image_mobile-applications_flavors_409_0.webp
    Разработка мобильного приложения для компании FLAVORS
    445

Нативная разработка Android на Kotlin

RecyclerView с DiffUtil.calculateDiff() на main thread, список из 500 элементов, средний Android-телефон 2021 года — и пользователь получает 200–400 мс фриза при каждом обновлении данных. Переносишь расчёт diff в фоновый поток через AsyncListDiffer — проблема исчезает. Такие вещи не очевидны без профилировщика и понимания threading-модели Android.

Kotlin + Jetpack Compose + Coroutines — это текущий production-стандарт для нативной Android-разработки. XML и View system никуда не исчезли, но новые проекты начинают на Compose.

Jetpack Compose: как работает рекомпозиция и почему это важно

Compose — декларативный UI-фреймворк. Вместо TextView.setText() и adapter.notifyItemChanged() — функции-composable, которые описывают UI как функцию состояния. При изменении состояния Compose перевычисляет только затронутые части дерева. Это называется рекомпозиция.

Проблема: рекомпозиция может быть слишком частой. Если передать в composable лямбду, созданную на каждой рекомпозиции родителя — дочерний composable будет рекомпозироваться каждый раз, даже если видимые данные не изменились.

// Плохо — новая лямбда на каждой рекомпозиции, дочерний компонент
// считает, что параметр изменился
@Composable
fun ParentScreen(viewModel: MyViewModel = hiltViewModel()) {
    val items by viewModel.items.collectAsState()
    ItemList(
        items = items,
        onItemClick = { id -> viewModel.selectItem(id) } // создаётся заново каждый раз
    )
}

// Хорошо — remember стабилизирует лямбду
@Composable
fun ParentScreen(viewModel: MyViewModel = hiltViewModel()) {
    val items by viewModel.items.collectAsState()
    val onItemClick = remember { { id: String -> viewModel.selectItem(id) } }
    ItemList(items = items, onItemClick = onItemClick)
}

Stability и @Stable/@Immutable

Compose определяет, нужно ли рекомпозировать composable, проверяя стабильность параметров. Тип считается стабильным, если Compose может гарантировать: если два значения равны по equals(), их UI-представление одинаково.

Примитивы, String, data-классы с val-полями стабильных типов — стабильны автоматически. List<T>нестабилен, потому что это интерфейс. MutableList может измениться без уведомления. Решение — ImmutableList из kotlinx.collections.immutable или @Immutable-аннотация на data-классе.

// List<Item> нестабилен — LazyColumn будет рекомпозироваться излишне
@Composable
fun ItemList(items: List<Item>) { ... }

// ImmutableList стабилен — Compose пропустит рекомпозицию если items не изменился
@Composable
fun ItemList(items: ImmutableList<Item>) { ... }

Для диагностики проблем рекомпозиции — Compose Compiler Metrics. В build.gradle добавляем флаги -P plugin:androidx.compose.compiler.plugins.kotlin:reportsDestination=... и получаем отчёт: какие composable restartable, какие skippable, почему параметр нестабилен.

LazyColumn и производительность списков

LazyColumn — эквивалент RecyclerView в Compose. key в items { } — обязательный параметр для любого списка, где элементы могут перемещаться или удаляться. Без key Compose не может отличить перемещение элемента от удаления одного и добавления другого, что ломает анимации и может вызвать неожиданное сбрасывание состояния ячейки.

LazyColumn {
    items(
        items = messages,
        key = { message -> message.id } // стабильный идентификатор
    ) { message ->
        MessageItem(message = message)
    }
}

contentType — дополнительная оптимизация. При наличии нескольких типов ячеек Compose может переиспользовать composition для ячеек одного типа. Это аналог getItemViewType в RecyclerView.

Kotlin Coroutines и архитектура

Coroutines — не просто «удобный способ писать async-код». Это структурированный concurrency с чётким scope и lifecycle.

viewModelScope — coroutine scope, привязанный к lifecycle ViewModel. Когда ViewModel очищается (onCleared()), все coroutines в scope автоматически отменяются. Это устраняет целый класс утечек, типичных для callback-based подхода.

@HiltViewModel
class OrderViewModel @Inject constructor(
    private val orderRepository: OrderRepository
) : ViewModel() {

    private val _uiState = MutableStateFlow<OrderUiState>(OrderUiState.Loading)
    val uiState: StateFlow<OrderUiState> = _uiState.asStateFlow()

    fun loadOrder(orderId: String) {
        viewModelScope.launch {
            _uiState.value = OrderUiState.Loading
            try {
                val order = orderRepository.getOrder(orderId) // suspend function
                _uiState.value = OrderUiState.Success(order)
            } catch (e: IOException) {
                _uiState.value = OrderUiState.Error(e.message)
            }
        }
    }
}

Flow vs LiveData

StateFlow и SharedFlow — рекомендованная замена LiveData на Kotlin-проектах. LiveData lifecycle-aware, но привязан к Android-платформе. Flow — чистый Kotlin, тестируется без Android-зависимостей.

collectAsState() в Compose подписывается на StateFlow и триггерит рекомпозицию при новом значении. lifecycleScope.launch { flow.collect { } } — для сборки в Fragment или Activity с учётом lifecycle через repeatOnLifecycle(Lifecycle.State.STARTED).

repeatOnLifecycle — это важно. Без него поток будет собираться даже когда приложение в фоне, что может приводить к обработке UI-событий, когда окно не активно.

Dispatcher и структурированный concurrency

Dispatchers.IO — для сетевых запросов и файловых операций. Dispatchers.Default — для CPU-intensive задач (парсинг, сортировка, шифрование). Dispatchers.Main — для UI.

withContext(Dispatchers.IO) переключает coroutine на нужный dispatcher без создания нового scope. Это эффективнее, чем launch(Dispatchers.IO) внутри другого launch.

// Правильный паттерн в Repository
suspend fun getOrders(): List<Order> = withContext(Dispatchers.IO) {
    orderDao.getAll() // Room автоматически suspend, но явный IO-dispatcher — хорошая практика
}

Hilt и dependency injection

Hilt — официальный DI-фреймворк для Android поверх Dagger 2. Устраняет boilerplate Dagger: не нужно писать Component и вручную связывать Module с Component.

@HiltViewModel + @Inject constructor — ViewModel с инъекцией зависимостей без фабрик. @Singleton, @ActivityScoped, @ViewModelScoped — правильный lifecycle для зависимостей.

Типичная ошибка: использовать @Singleton для репозитория, который держит контекст Activity. Это утечка Activity. Правило: @Singleton только для зависимостей, которым нужен Application context или которые не хранят Android-специфичное состояние.

WorkManager и фоновые задачи

WorkManager — для гарантированных фоновых задач, которые должны выполниться даже после перезапуска приложения или устройства. Синхронизация данных, отправка аналитики, загрузка файлов.

CoroutineWorker — suspend-версия Worker. Работает в Dispatchers.IO по умолчанию.

Android 14 ужесточил требования к фоновому выполнению. FOREGROUND_SERVICE_TYPE стал обязательным для foreground services. WorkManager правильно обрабатывает ограничения (сеть, зарядка) и не требует foreground service для большинства задач.

Инструменты

Android Studio Profiler. CPU profiler с System Trace — видно всё: coroutine suspension points, RenderThread, MainThread. Memory profiler — heap dump, allocation tracking. Network profiler — все HTTP-запросы с телами.

Compose Layout Inspector. Дерево composable с указанием recomposition count. Видно, какие composable рекомпозируются слишком часто — точнее любого логирования.

LeakCanary. Автоматическое обнаружение утечек памяти в development-сборке. Показывает reference chain до утечки. Добавляется одной зависимостью, работает без конфигурации.

Firebase Crashlytics + Performance Monitoring. Crash-free rate по версиям, Network request traces, Custom traces для критичных операций.

Сроки

Сложность Ориентировочный срок
MVP (6–10 экранов, REST API) 6–10 недель
Среднее приложение (20–30 экранов) 3–5 месяцев
Сложное (платежи, ML Kit, Compose + custom UI) 5–9 месяцев

Стоимость рассчитывается после анализа требований и ТЗ.