Карты и геолокация в мобильных приложениях: Google Maps, MapKit, геофенсинг, трекинг
Геолокация в мобильных приложениях — это не «добавить карту». Это работа с разрешениями, точностью, энергопотреблением и поведением ОС, которое принципиально различается на iOS и Android. Трекер доставки, приложение для бега, карта магазинов — каждый случай требует разного подхода к тому, как и когда запрашивать координаты.
Разрешения: один из самых частых источников плохих отзывов
На iOS разрешение на геолокацию — самое чувствительное после микрофона и камеры. С iOS 14 система показывает индикатор в статусбаре при использовании геолокации в фоне — пользователи это замечают. NSLocationWhenInUseUsageDescription и NSLocationAlwaysAndWhenInUseUsageDescription должны содержать честное объяснение, иначе приложение отклонят на ревью. Запрашивать always разрешение сразу при старте — верный способ получить отказ от большинства пользователей. Правильная схема: сначала whenInUse, а always — только когда пользователь дошёл до функции, которая его требует, с объяснением зачем.
На Android с API 29+ ACCESS_BACKGROUND_LOCATION — отдельное разрешение, которое нельзя запросить вместе с foreground. Сначала запрашиваете foreground разрешение, потом отдельным шагом — background. Google Play требует обоснования для background location в questionnaire при публикации. Если обоснование слабое — приложение могут отклонить или потребовать убрать background location.
Точность и энергопотребление: как не разряжать аккумулятор
Постоянный GPS на максимальной точности потребляет 100–150 мВт — аккумулятор садится за 4–6 часов. Для большинства задач это избыточно.
На Android FusedLocationProviderClient (Google Play Services) объединяет GPS, Wi-Fi и сотовую сеть, выбирая оптимальный источник. LocationRequest.Builder с приоритетами:
-
PRIORITY_HIGH_ACCURACY— GPS включён, для навигации -
PRIORITY_BALANCED_POWER_ACCURACY— точность ~100 метров, Wi-Fi + сотовая -
PRIORITY_LOW_POWER— точность ~10 км, только сотовая -
PRIORITY_PASSIVE— координаты от других приложений, без активного запроса
Для трекера пробежки в активном режиме — HIGH_ACCURACY с интервалом 2–5 секунд. Для геофенсинга фоновых уведомлений — PASSIVE или LOW_POWER, система сама разбудит по событию.
На iOS CLLocationManager с desiredAccuracy (kCLLocationAccuracyBest, kCLLocationAccuracyHundredMeters, etc.) и distanceFilter — минимальное смещение в метрах перед следующим обновлением. Для трекинга маршрута с сохранением батареи: desiredAccuracy = kCLLocationAccuracyNearestTenMeters, distanceFilter = 10 — получаем обновления только при реальном движении.
Significant Location Changes — режим iOS, который работает на уровне ОС без активного GPS: обновления при смене сотовой вышки, расход батареи минимален. Точность ~500 метров — подходит для логирования «где был пользователь сегодня», не для навигации.
Google Maps SDK vs MapKit vs Mapbox vs OpenStreetMap
| SDK | Платформа | Офлайн-карты | Кастомный стиль | Без Google Services |
|---|---|---|---|---|
| Google Maps SDK | iOS/Android | Нет (только Maps API) | Да (Cloud-based) | Нет |
| MapKit | iOS | Нет | Ограниченно | Да |
| Mapbox Maps | iOS/Android | Да | Полностью | Да |
| HERE Maps | iOS/Android | Да | Да | Да |
| OpenStreetMap + MapLibre | iOS/Android/Flutter | Да | Полностью | Да |
Google Maps SDK — выбор по умолчанию для большинства проектов: знакомый UI, хорошая документация, Directions API, Places Autocomplete. Ограничение — зависимость от Google Play Services (проблема для Huawei) и ценовая политика при высоких объёмах запросов.
Mapbox предпочтителен, когда нужен кастомный стиль карты (корпоративный брендинг, тёмная тема), офлайн-карты для работы без сети, или доступность на устройствах без GMS. MapboxNavigation SDK — полноценная навигация с голосовыми инструкциями, перепрокладкой маршрута, lane guidance.
Для Flutter — google_maps_flutter (официальный), flutter_map (OpenStreetMap + MapLibre, полностью open-source), mapbox_maps_flutter (с 2023 года официальный Mapbox SDK).
Геофенсинг: уведомления по зоне
Геофенсинг — запуск события при входе/выходе из географической зоны (круг заданного радиуса).
На Android — GeofencingClient из Google Location Services. Добавляем Geofence объекты с setTransitionTypes(GEOFENCE_TRANSITION_ENTER | GEOFENCE_TRANSITION_EXIT) и PendingIntent на BroadcastReceiver. Ограничение: максимум 100 активных геофенсов на приложение, минимальный радиус ~150 метров (на практике из-за точности), срабатывание с задержкой до нескольких минут в целях экономии батареи.
На iOS — CLCircularRegion + CLLocationManager.startMonitoring(for:). Лимит: 20 регионов на приложение. ОС управляет когда проверять — разработчик не контролирует задержку. Для более точного геофенсинга с маленьким радиусом — iBeacon (CLBeaconRegion) или CLVisit для мест, где пользователь провёл время.
Если нужно больше 20 (iOS) или 100 (Android) зон — нужна серверная логика: периодически отправляем координаты на сервер, сервер проверяет попадание в зоны и отправляет push. Менее точно по времени, но масштабируется на тысячи зон.
Трекинг маршрутов и фоновая геолокация
Трекинг маршрута пробежки или маршрута курьера в фоне — технически разные задачи.
На iOS фоновая геолокация работает через UIBackgroundModes: location в Info.plist. Без этого ключа при уходе приложения в фон CLLocationManager получает несколько минут и засыпает. С ключом — работает постоянно, но система может приостановить при критически низком заряде.
Для трекера пробежки на iOS паттерн: startUpdatingLocation при старте тренировки, координаты пишем в Core Data каждые 5 секунд, на паузе — stopUpdatingLocation, но оставляем startMonitoringSignificantLocationChanges чтобы приложение не «потерялось» совсем.
На Android для курьерского трекинга нужен Foreground Service с FOREGROUND_SERVICE_TYPE_LOCATION (обязательно с API 29). Foreground service показывает постоянное уведомление — это требование платформы, не баг. Без него Android Doze убьёт обновления геолокации. WorkManager для фоновых задач здесь не подходит — он не гарантирует непрерывность.
Алгоритмическая часть трекинга маршрута: сырые GPS-координаты зашумлены. Для сглаживания — алгоритм Ramer-Douglas-Peucker для упрощения трека или Kalman Filter для фильтрации шума в реальном времени. Без фильтрации трек выглядит как случайные зигзаги, а расчётное расстояние на 20–30% больше реального.
Срок и состав работ
Базовая интеграция карты с маркерами и поиском адресов — 1–2 недели. Геофенсинг с push-уведомлениями — 2–3 недели с учётом тестирования на реальных устройствах (эмулятор плохо симулирует геофенсинг-задержки). Полноценный трекинг маршрутов с фоновым режимом, сглаживанием и серверной синхронизацией — 4–6 недель.







