Android Good Reads
4.1K subscribers
383 photos
16 videos
1 file
2.15K links
Самые интересные статьи, видео и новости, связанные с Android разработкой. Не больше трёх материалов в день.

Автор канала: @Lamprof

Размещение рекламы: @tanyasanovna
Download Telegram
Рабочая группа по продвижению использования Swift в android разработке

Основная идея - сделать Android официально поддерживаемой платформой языка Swift. Задачи перед собой поставили стандартные для кроссплатформенного фреймворка, обсуждение открытое, так что можете поучаствовать.

А чтобы быть готовым - пойду подпишусь на iOS Good Reads (@iosgr)
🥴247👍4🔥4😁4
Недавний релиз Kotlin 2.2.0

👉 Фича в превью: контекстные параметры
👉 Экспериментальные функции теперь стабильны: guard conditions, нелокальные операторы break и continue, а также multi-dollar интерполяция строк.
👉 Унифицированное управление предупреждениями компилятора.
👉 Изменения в генерации методов по умолчанию для функций в интерфейсах.
👉 Поддержка LLVM 19 и новые возможности для отслеживания и настройки потребления памяти
👉 Разделение Wasm-таргета и возможность настраивать Binaryen для каждого проекта
👉 Исправлена работа метода copy, генерируемого для интерфейсов с аннотацией @JsPlainObject
👉 В плагин Kotlin Gradle включена проверка бинарной совместимости.
👉 API для Base64 и HexFormat теперь стабильны.

10 июля в 16:00 UTC будет стрим с разработчиками
👍7🔥54
Уменьшаем размер приложения

Вам стоит пропустить эту статью если у вас уже настроено:
👉 Минимальный R8 со специфичными правилами под каждую библиотеку. Вы проверяете его работу не на проде, а с помощью testProguardFiles, например
👉 Перевели все картинки в webp формат, вырезали ненужные ресурсы в конфигурациях
👉 Аккуратно переиспользуете layout
👉 Используете appBundle
👍62🔥1
Шпаргалка по flow операторам

Внутри примеры кода с пояснением для:
👉 map, filter, take(n)
👉 reduce, fold, runningReduce/scan,
👉 transform
👉 FlatMapConcat, FlatMapMerge, FlatMapLatest
👉 flowOn, buffer, conflate, collectLatest
👉 zip, combine, merge
👉 catch, onCompletion, retryWhen
👉 onEach, debounce(timeoutMillis), distinctUntilChanged

Проверяем себя, что все вышеперечисленное использовали хоть раз и знаем!
👍15🔥42
Так как я чертовски заинтересован в увеличении комьюнити CMP разработки, то вот вам свежая и полезная библиотечка Alarmee, которая является оберткой для всякого рода пушей.
Firebase, Локальные, Отложенные и повторяющиеся пуши - всё тут!
Никаких больше болей по надстройке пушей на платформах!
10👍7🔥3👎1
Про поддержку 16KB page

Зачем?
Девайсы с поддержкой 16 KB работают быстрее на 5-10%, а так же:
👉 Старт приложений быстрее (до 30% для некоторых приложений, 3.16% в среднем),
👉 Меньше расход батареи (на 4.56% меньше)
👉 Быстрее запуск камеры (4.48-6.60%)

Как?
👉APK Analyze/Alignment Checks/Lint поможет найти вселомающую зависимость
👉Обновляем AGP до 8.5.1+, NDK до r28+.
👉Если не помогло то кропотливо выискиваем нужные флаги из официальных рекомендаций

Тестируем
Необходимые эмуляторы уже в Android Studio
👍14🔥43👏1🏆1
Напоминание: В пору кучи AI - сгенерированного кода не забывайте закручивать гайки через lint / detekt, дабы снимать с себя большую часть работ по ревью этого в проекте. Гайд как это сделать

Лучше, конечно, обратиться к официальной доке: https://detekt.dev/docs/intro
🔥9👍7
Изучаем PausableComposition в Jetpack Compose

Введён недавно, в обновлении 1.9.х. Вообще это внутреннее апи и если у вас нет желания разбираться в том как Compose оптимизирует производительность, то это знание вам и не потребуется, но интересно же?
🔥5
История о том как с помощью Perfetto нашли и решили проблему долгого ответа от подключенных USB девайсов к Android платформе. Пришлось сделать пул в AOSP, но задержка упала на 40%.

Если вы разрабатываете кассовое ПО - это точно статья, обязательная к прочтению
🔥6👍5👌2
Android Good Reads
Я не хочу превращать Android канал в новостник на тему очередного прорыва в AI, поэтому соберу в кучу несколько полезных статей и холиварных топиков, которые отражают мою позицию. 👉 Как AI влияет на способность думать "гораздо проще написать промпт, закинуть…
Продолжаем искать полезности в эре AI для Android:

👉 Сгенерированная вики для любого репозитория. Пример для compose-mp репы и для retrofit.
👉 Junie, интегрированная в IDE, получила необходимый набор инструментов для современного AI-coding плагина, а так же возможность взаимодействовать через Github
👉 SDD (Spec driven development) Не новый подход, нашедший свое применение в AI инструментах. В кратце: сначала определяем задачу и все требования (spec) -> затем ее дизайним -> и только потом приступаем к реализации.

Лично я по чуть-чуть перешел в терминальные тулзы(ClaudeCode, GeminiCLI, OpenCode), потому что переключаться между Android Studio и форками VScode уже порядком надоело
👍9🔥42
Google Sans Code

Monospace шрифт с различным написанием "l" и "1", or "O" и "0". Уже доступен в Android Studio. Выглядит удобно
🔥12👍3
Нынче в Андроиде

👉 Canary версии обновлений для девайсов. На самой ранней стадии будем ловить все проблемы будущей операционки
👉 Продолжают вкладываться в Agentic AI внутри Android Studio. Могу дать только крайне негативный отзыв. Лучше бы вложились в Junie
👉 Еще разок о 16KB page size. Разбирались с вами вот тут
👉 Обновление для one-time product внутри Google Play Billing
👉 Грядущий изменения в разработки приложений под часы. Со следующего года буду введены ограничения на публикацию устаревших версий
👉 Google Play Игры теперь и на PC. Пример на основе Mecha BREAK.
👍4😁2
Очень специфичный кейс. Отправляем RecyclerView ивенты в родительский Compose.

Возможный кейс при миграции на compose и сильной атомарности компонентов. В целом важен только код внутри статьи, но идея заключается в композиции NestedScrollConnection и NestedScrollDispatcher
🤯3👍2
Вчера был релиз Kotlin 2.2.20-RC

👉 В основном упор на мультиплатформу. Минорные изменение и стабилизация
👉 Можно в экспериментальном режиме посмотреть фичи из 2.3.0

Перегрузка suspend в лямде

fun transform(block: () -> Int) {}
fun transform(block: suspend () -> Int) {}


return в expression bodies с explicit return types. На примере более очевидно. Больше примеров тут

fun returnInsideIf() = when {
else -> {
val result = if (someCondition()) return "" else "value"
result
}
}
4👍4🔥2
Эволюция архитектур в Android

⬇️ Мономодуль. Весь код в 1 модуле, код не переиспользуется между приложениями
⬇️ Упрощенная модуляризация. Общая часть выносится в единый base модуль. Все остальные модули зависят от него.
⬇️ Полная модуляризация. Помимо общего kotlin/android модуля появляется data/domain/presentation модули и сложный менеджмент зависимостей меж фиче-модулями
⬇️ После того как вы обнаружили себя в миллионе модулей со временем сборки, стремящимся в бесконечность, - начинается объединение в фичегруппы и уменьшение количество модулей.

Подробнее про каждую внутри статьи
На какой стадии ваш проект по этой линейке?
👍82🤔2
Kotlin/Compose MP - Что будет дальше?

👉 Фокус на iOS/web. Очень ждем, особенно первое. Зависимости и сборка это боль
👉 Улучшения IDE для MP разработки. Новый плагин это большая беда. Много багов, сломали что работало. Надеюсь будет лучше
👉 Kotlin Web/Wasm хотят выпустить в бету
👉 Ну и как обычно. Сборка быстрее, конфигурирование проще в грядущих версиях

Следим!
👍6🔥54
Android Good Reads
Rich Errors в Kotlin 2.4 Анонсировали на KotlinConf. На мой взгляд, классная обертка на замену старому-доброму try-catch. Какие возможные проблемы видите в этом?
Анонсированные Rich Errors все ближе

В статье о том как к этому подготовиться, ну а я предлагаю взглянуть на side-by-side пример:

Было

fun loadConfig(path: Path): Result<Config> =
runCatching { fileSystem.readBytes(path) }
.mapCatching { bytes -> parseConfig(bytes) }
.recover { e ->
if (e is NoSuchFileException) defaultConfig() else throw e
}

fun useConfig(path: Path) {
loadConfig(path)
.onSuccess { cfg -> startApp(cfg) }
.onFailure { e ->
when (e) {
is NoSuchFileException -> showMissingConfigWarning()
is ConfigFormatException -> showConfigParseError(e)
else -> showGenericError(e)
}
}
}


Стало

// Rich errors
// Define error variants (could be data objects/classes in your domain)
data object IoFailure
data class ParseFailure(val details: String)

fun loadConfig(path: Path): Config | IoFailure | ParseFailure

fun useConfigRich(path: Path) {
when (val r = loadConfig(path)) {
is Config -> startApp(r)
is IoFailure -> showMissingConfigWarning()
is ParseFailure -> showConfigParseErrorMessage(r.details)
}
}
🔥174🥰4