Кроссплатформенная разработка: Flutter, React Native, KMM
Стартап хочет два приложения — iOS и Android — с бюджетом на одну команду. Или корпорация хочет выпустить внутренний инструмент за 3 месяца на обе платформы. Кроссплатформенная разработка решает конкретную экономическую задачу: один кодовый базис против двух. Вопрос не «кроссплатформа или нативная» — вопрос «какой инструмент под какую задачу».
Три главных игрока сейчас: Flutter, React Native и Kotlin Multiplatform Mobile. Они решают разные проблемы и плохо сравниваются в лоб.
Flutter vs React Native: детальный разбор
Это сравнение, которое реально влияет на выбор. Разберём по конкретным техническим характеристикам.
Модель рендеринга
Flutter рендерит UI самостоятельно через Impeller (заменил Skia с Flutter 3.10). Платформа предоставляет только canvas — Flutter рисует каждый пиксель сам. Это означает:
- Пиксельная точность на всех платформах. Один и тот же виджет выглядит одинаково на iOS и Android — хорошо для брендовых приложений, плохо если нужно выглядеть «нативно» на каждой платформе.
- Нет зависимости от версии ОС. Material 3 в Flutter работает одинаково на Android 8 и Android 14. Системные компоненты Android не участвуют.
-
Platform channels для нативного кода. Доступ к камере, Bluetooth, NFC — через
MethodChannelилиEventChannel.flutter_camera,flutter_blue_plus— это обёртки над platform channels.
React Native использует нативные компоненты платформы. <View> на iOS — это UIView. <Text> — это UILabel. Это означает:
- Нативный look and feel без дополнительных усилий.
- New Architecture (Fabric + TurboModules) с JSI убрала JSON-мост между JS и нативным кодом. Синхронные вызовы работают без сериализации. Это критично для анимаций и жестов.
- React Native Reanimated 3 запускает worklets на UI thread — анимации в 60/120 fps без блокировки JS thread.
Производительность на практике
Для большинства бизнес-приложений разница в производительности между Flutter и React Native New Architecture незаметна пользователю. Разница появляется в граничных случаях.
Flutter медленнее при взаимодействии с платформенными API через platform channels — каждый вызов асинхронный, есть накладные расходы на сериализацию данных. google_maps_flutter рендерит карту через PlatformView — это нативный UIView/View, встроенный в Flutter-дерево. До Impeller это вызывало проблемы с производительностью (Hybrid Composition vs Virtual Display). С Impeller ситуация улучшилась, но PlatformView по-прежнему дороже чистого Flutter-виджета.
React Native медленнее в сценариях с тяжёлой JS-логикой на main thread. Парсинг большого JSON, сложные вычисления — это блокирует JS thread и видно как фриз UI. Решение: Hermes (JS-движок, оптимизированный для RN) + вынос вычислений в нативный модуль или react-native-workers.
Экосистема и зрелость
| Параметр | Flutter | React Native |
|---|---|---|
| Язык | Dart | JavaScript / TypeScript |
| Пакетный менеджер | pub.dev | npm / yarn |
| Крупные компании | Google, Alibaba, BMW | Meta, Microsoft, Shopify |
| Hot reload | Да (stateful) | Да (Fast Refresh) |
| Desktop (macOS, Windows) | Да (stable) | Экспериментально |
| Web | Да (CanvasKit / HTML) | Частично (через React) |
| Размер APK/IPA | ~6 MB базовый | ~4 MB базовый |
Dart — барьер входа для команд с JS/TS-бэкграундом. Освоить базовый Dart за неделю реально. Перестроить мышление под Flutter-виджеты и widget tree — занимает больше.
TypeScript в React Native — стандарт де-факто. Команда с React-опытом начинает работать продуктивно быстрее.
Когда выбирать Flutter
- Нужен единый брендовый UI на всех платформах (iOS, Android, Web, Desktop)
- Команда готова к Dart
- Много кастомной анимации и кастомного UI — Flutter более предсказуем
- Приложение не завязано на специфические нативные API
Когда выбирать React Native
- Команда с React/TypeScript-экспертизой
- Нужен нативный look and feel платформы
- Активное использование нативных компонентов (Maps, Camera с нативными возможностями)
- Шаринг кода с React-вебом через монорепозиторий
Kotlin Multiplatform Mobile: другая история
KMM решает не UI-проблему, а проблему дублирования бизнес-логики. Концепция: пишем бизнес-логику, работу с сетью, кеш, валидацию один раз на Kotlin. iOS получает .framework через Kotlin/Native, Android использует библиотеку напрямую. UI на каждой платформе — нативный.
// Shared Kotlin код — работает на iOS и Android
class UserRepository(
private val httpClient: HttpClient, // Ktor
private val database: AppDatabase // SQLDelight
) {
suspend fun getUser(id: String): User {
return database.userQueries.selectById(id).executeAsOneOrNull()
?: httpClient.get("$BASE_URL/users/$id").body<User>().also {
database.userQueries.insert(it)
}
}
}
Ktor — HTTP-клиент для KMM (работает на iOS через Darwin engine, на Android через OkHttp). SQLDelight генерирует typesafe Kotlin API для SQLite, работает на обеих платформах.
Реальные ограничения KMM
Coroutines на iOS. Kotlin Coroutines работают, но с оговорками. suspend-функции из shared кода вызываются с iOS через автоматически сгенерированные обёртки. SKIE (Swift/Kotlin Interface Enhancer) от Touchlab значительно улучшает Swift-интерфейс: async/await вместо callback, AsyncStream для Flow. Без SKIE работать с coroutines из Swift неудобно.
Compose Multiplatform. JetBrains развивает Compose для iOS — UI на Compose работает на iOS через Metal. Это стирает границу с Flutter: один Compose-код для обеих платформ. Статус в 2024: Beta, в production есть ранние adopters (Touchlab, JetBrains собственные продукты), но стабильность ниже Flutter.
Сложность iOS-интеграции. XCFramework из KMM-модуля добавляется в Xcode-проект. SPM-интеграция появилась и работает. Но iOS-разработчики должны понимать Kotlin-API и правила работы с памятью через Kotlin/Native (ARC + Kotlin GC работают вместе, это не всегда очевидно).
Когда KMM оправдан
У компании уже есть зрелые iOS и Android команды, которые дублируют бизнес-логику. Переводить всё на Flutter или React Native — слишком радикально. KMM позволяет начать с малого: вынести сетевой слой и модели в shared, оставить UI нативным. Постепенная миграция без переписывания всего.
Типичные ошибки при выборе технологии
Выбор Flutter «потому что один кодовый базис» для приложения, которое сильно завязано на нативные API (custom camera, BLE, background processing). Реализовывать это через platform channels — дополнительная сложность, которая съедает выигрыш по скорости разработки.
React Native без понимания JS-thread. Тяжёлые операции на JS thread дают видимые фризы. Это решаемо, но требует понимания architecture — иначе приложение будет работать хуже нативного.
KMM без iOS-разработчика в команде. Shared Kotlin-код требует iOS-инженера, который интегрирует фреймворк в Xcode, пишет SwiftUI поверх KMM API и дебажит Kotlin/Native-краши.
Процесс и сроки
Кроссплатформенный проект проходит те же этапы, что нативный: аудит требований → выбор стека → дизайн → разработка → тестирование на реальных устройствах обеих платформ → публикация в App Store и Google Play → поддержка.
Тестирование на реальных устройствах — не опциональный этап. Эмулятор не воспроизводит проблемы с памятью на бюджетных Android-телефонах и не показывает разницу в поведении жестов на iOS.
| Тип проекта | Flutter | React Native |
|---|---|---|
| MVP (8–12 экранов) | 7–12 недель | 7–12 недель |
| Среднее (20–30 экранов) | 3–5 месяцев | 3–5 месяцев |
| Сложное (нативные интеграции, AI) | 5–8 месяцев | 5–8 месяцев |
Стоимость — индивидуально после анализа стека и требований.







