Как и предыдущие выпуски, Android 15 включает изменения в поведении, которые могут повлиять на ваше приложение. Следующие изменения в поведении применяются исключительно к приложениям, ориентированным на Android 15 или более позднюю версию. Если ваше приложение ориентировано на Android 15 или более поздней версии, вам следует изменить его так, чтобы оно правильно поддерживало такое поведение, если это применимо.
Обязательно ознакомьтесь со списком изменений поведения, которые влияют на все приложения, работающие на Android 15, независимо от targetSdkVersion
вашего приложения.
Основная функциональность
Android 15 изменяет или расширяет различные основные возможности системы Android.
Изменения в службах переднего плана
Мы вносим следующие изменения в службы переднего плана в Android 15.
- Поведение времени ожидания службы синхронизации данных на переднем плане
- Новый тип приоритетной службы обработки мультимедиа
- Ограничения на широковещательные приемники
BOOT_COMPLETED
, запускающие службы приоритета - Ограничения на запуск служб переднего плана, пока приложение имеет разрешение
SYSTEM_ALERT_WINDOW
Поведение тайм-аута службы синхронизации данных на переднем плане
В Android 15 представлен новый режим тайм-аута для dataSync
для приложений, предназначенных для Android 15 (уровень API 35) или выше. Это поведение также применимо к новому типу службы переднего плана mediaProcessing
.
Система разрешает службам dataSync
приложения работать в общей сложности 6 часов в течение 24-часового периода, после чего система вызывает метод Service.onTimeout(int, int)
работающей службы (представленный в Android 15). В это время у службы есть несколько секунд для вызова Service.stopSelf()
. При вызове Service.onTimeout()
служба больше не считается службой переднего плана. Если служба не вызывает Service.stopSelf()
, система выдает внутреннее исключение. Исключение регистрируется в Logcat со следующим сообщением:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type dataSync did not stop within its timeout: [component name]"
Чтобы избежать проблем с этим изменением поведения, вы можете выполнить одно или несколько из следующих действий:
- Пусть ваш сервис реализует новый метод
Service.onTimeout(int, int)
. Когда ваше приложение получит обратный вызов, обязательно вызовитеstopSelf()
в течение нескольких секунд. (Если вы не остановите приложение сразу, система выдаст ошибку.) - Убедитесь, что службы
dataSync
вашего приложения не работают более 6 часов в течение 24-часового периода (если только пользователь не взаимодействует с приложением, сбрасывая таймер). - Запускайте службы
dataSync
переднего плана только в результате прямого взаимодействия с пользователем; поскольку ваше приложение находится на переднем плане при запуске службы, у вашей службы есть полные шесть часов после того, как приложение перейдет в фоновый режим. - Вместо использования службы переднего плана
dataSync
используйте альтернативный API .
Если службы приоритетного плана dataSync
вашего приложения работали в течение 6 часов за последние 24 часа, вы не сможете запустить другую службу приоритетного плана dataSync
, пока пользователь не переведет ваше приложение на передний план (что сбрасывает таймер). Если вы попытаетесь запустить другую службу переднего плана dataSync
, система выдаст исключение ForegroundServiceStartNotAllowedException
с сообщением об ошибке, например «Ограничение времени для типа службы переднего плана dataSync уже исчерпано».
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить таймауты синхронизации данных, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить таймауты, выполните следующую команду adb
:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
Вы также можете настроить период ожидания, чтобы было проще протестировать поведение вашего приложения при достижении предела. Чтобы установить новый период ожидания, выполните следующую команду adb
:
adb shell device_config put activity_manager data_sync_fgs_timeout_duration duration-in-milliseconds
Новый тип приоритетной службы обработки мультимедиа
В Android 15 представлен новый тип службы переднего плана — mediaProcessing
. Этот тип службы подходит для таких операций, как перекодирование медиафайлов. Например, мультимедийное приложение может загрузить аудиофайл, и ему необходимо преобразовать его в другой формат перед воспроизведением. Вы можете использовать службу переднего плана mediaProcessing
чтобы гарантировать, что преобразование продолжается, даже когда приложение находится в фоновом режиме.
Система разрешает службам mediaProcessing
приложения работать в общей сложности 6 часов в течение 24-часового периода, после чего система вызывает метод Service.onTimeout(int, int)
работающей службы (представленный в Android 15). В это время у службы есть несколько секунд для вызова Service.stopSelf()
. Если служба не вызывает Service.stopSelf()
, система выдает внутреннее исключение. Исключение регистрируется в Logcat со следующим сообщением:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type mediaProcessing did not stop within its timeout: [component name]"
Чтобы избежать исключения, вы можете сделать одно из следующих действий:
- Пусть ваш сервис реализует новый метод
Service.onTimeout(int, int)
. Когда ваше приложение получит обратный вызов, обязательно вызовитеstopSelf()
в течение нескольких секунд. (Если вы не остановите приложение сразу, система выдаст ошибку.) - Убедитесь, что службы
mediaProcessing
вашего приложения не работают более 6 часов в течение любого 24-часового периода (если только пользователь не взаимодействует с приложением, сбрасывая таймер). - Запускайте службы переднего плана
mediaProcessing
только в результате прямого взаимодействия с пользователем; поскольку ваше приложение находится на переднем плане при запуске службы, у вашей службы есть полные шесть часов после того, как приложение перейдет в фоновый режим. - Вместо использования службы переднего плана
mediaProcessing
используйте альтернативный API , например WorkManager.
Если службы переднего плана mediaProcessing
вашего приложения работали в течение 6 часов за последние 24, вы не сможете запустить другую службу переднего плана mediaProcessing
, пока пользователь не переведет ваше приложение на передний план (что сбрасывает таймер). Если вы попытаетесь запустить другую службу переднего плана mediaProcessing
, система выдаст ForegroundServiceStartNotAllowedException
с сообщением об ошибке, например «Ограничение времени уже исчерпано для типа службы переднего плана mediaProcessing».
Дополнительные сведения о типе службы mediaProcessing
см. в разделе Изменения в типах служб переднего плана для Android 15: обработка мультимедиа .
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить тайм-ауты обработки мультимедиа, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить таймауты, выполните следующую команду adb
:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
Вы также можете настроить период ожидания, чтобы было проще протестировать поведение вашего приложения при достижении предела. Чтобы установить новый период ожидания, выполните следующую команду adb
:
adb shell device_config put activity_manager media_processing_fgs_timeout_duration duration-in-milliseconds
Ограничения на широковещательные приемники BOOT_COMPLETED
, запускающие службы приоритета
在启动 BOOT_COMPLETED
广播接收器方面存在新限制
前台服务。BOOT_COMPLETED
接收器不能启动
以下类型的前台服务:
dataSync
camera
mediaPlayback
phoneCall
mediaProjection
microphone
(自 Android 14 起,microphone
就受到此限制)
如果 BOOT_COMPLETED
接收器尝试启动任何上述类型的前台
服务,系统会抛出 ForegroundServiceStartNotAllowedException
。
测试
如需测试应用的行为,您可以启用这些新限制,即使您的应用并未以 Android 15 为目标平台(只要应用在 Android 15 设备上运行)也是如此。运行以下 adb
命令:
adb shell am compat enable FGS_BOOT_COMPLETED_RESTRICTIONS your-package-name
如需在不重启设备的情况下发送 BOOT_COMPLETED
广播,请运行以下 adb
命令:
adb shell am broadcast -a android.intent.action.BOOT_COMPLETED your-package-name
Ограничения на запуск служб переднего плана, пока приложение имеет разрешение SYSTEM_ALERT_WINDOW
Раньше, если приложение имело разрешение SYSTEM_ALERT_WINDOW
, оно могло запустить службу переднего плана, даже если приложение в данный момент находилось в фоновом режиме (как обсуждалось в разделе об исключениях из ограничений фонового запуска ).
Если приложение предназначено для Android 15, это исключение теперь уже. Теперь приложению необходимо иметь разрешение SYSTEM_ALERT_WINDOW
, а также иметь видимое окно наложения. То есть приложению необходимо сначала запустить окно TYPE_APPLICATION_OVERLAY
, и это окно должно быть видимым перед запуском службы переднего плана.
Если ваше приложение пытается запустить службу переднего плана в фоновом режиме, не отвечая этим новым требованиям (и у него нет других исключений), система выдает ForegroundServiceStartNotAllowedException
.
Если ваше приложение объявляет разрешение SYSTEM_ALERT_WINDOW
и запускает службы переднего плана в фоновом режиме, это изменение может повлиять на него. Если ваше приложение получает исключение ForegroundServiceStartNotAllowedException
, проверьте порядок операций вашего приложения и убедитесь, что в нем уже есть активное окно наложения, прежде чем оно попытается запустить службу переднего плана из фона. Вы можете проверить, видимо ли ваше окно наложения в данный момент, вызвав View.getWindowVisibility()
, или вы можете переопределить View.onWindowVisibilityChanged()
чтобы получать уведомления при каждом изменении видимости.
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить эти новые ограничения, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить эти новые ограничения на запуск служб переднего плана в фоновом режиме, выполните следующую команду adb
:
adb shell am compat enable FGS_SAW_RESTRICTIONS your-package-name
Изменения в том, когда приложения могут изменять глобальное состояние режима «Не беспокоить».
Приложения, предназначенные для Android 15 (уровень API 35) и выше, больше не могут изменять глобальное состояние или политику режима «Не беспокоить» (DND) на устройстве (путем изменения пользовательских настроек или отключения режима «Не беспокоить»). Вместо этого приложения должны внести AutomaticZenRule
, которое система объединяет в глобальную политику с существующей схемой наиболее ограничительной политики. Вызовы к существующим API, которые ранее влияли на глобальное состояние ( setInterruptionFilter
, setNotificationPolicy
), приводят к созданию или обновлению неявного AutomaticZenRule
, который включается и выключается в зависимости от цикла вызовов этих вызовов API.
Обратите внимание, что это изменение влияет на наблюдаемое поведение только в том случае, если приложение вызывает setInterruptionFilter(INTERRUPTION_FILTER_ALL)
и ожидает, что этот вызов деактивирует AutomaticZenRule
, который ранее был активирован его владельцами.
Изменения API OpenJDK
В Android 15 продолжается работа по обновлению основных библиотек Android, чтобы они соответствовали функциям последних выпусков OpenJDK LTS.
Некоторые из этих изменений могут повлиять на совместимость приложений, предназначенных для Android 15 (уровень API 35):
Изменения в API форматирования строк : проверка индекса аргумента, флагов, ширины и точности теперь более строгая при использовании следующих API
String.format()
иFormatter.format()
:-
String.format(String, Object[])
-
String.format(Locale, String, Object[])
-
Formatter.format(String, Object[])
-
Formatter.format(Locale, String, Object[])
Например, следующее исключение выдается, когда используется индекс аргумента 0 (
%0
в строке формата):IllegalFormatArgumentIndexException: Illegal format argument index = 0
В этом случае проблему можно решить, используя индекс аргумента 1 (
%1
в строке формата).-
Изменения в типе компонента
Arrays.asList(...).toArray()
: при использованииArrays.asList(...).toArray()
типом компонента результирующего массива теперь являетсяObject
, а не тип элементы базового массива. Итак, следующий код генерирует исключениеClassCastException
:String[] elements = (String[]) Arrays.asList("one", "two").toArray();
В этом случае, чтобы сохранить
String
как тип компонента в результирующем массиве, вы можете вместо этого использоватьCollection.toArray(Object[])
:String[] elements = Arrays.asList("two", "one").toArray(new String[0]);
Изменения в обработке языковых кодов . При использовании
Locale
API языковые коды для иврита, идиша и индонезийского языка больше не преобразуются в их устаревшие формы (иврит:iw
, идиш:ji
и индонезийский:in
). При указании кода языка для одного из этих языков вместо этого используйте коды из ISO 639-1 (иврит:he
, идиш:yi
и индонезийский:id
).Изменения в случайных последовательностях целых чисел : после изменений, внесенных в https://bugs.openjdk.org/browse/JDK-8301574 , следующие методы
Random.ints()
теперь возвращают последовательность чисел, отличную от методовRandom.nextInt()
. делать:Как правило, это изменение не должно приводить к нарушению работы приложения, но ваш код не должен ожидать, что последовательность, сгенерированная методами
Random.ints()
будет соответствоватьRandom.nextInt()
.
Новый API SequencedCollection
может повлиять на совместимость вашего приложения после обновления compileSdk
в конфигурации сборки вашего приложения для использования Android 15 (уровень API 35) :
Конфликт с функциями расширения
MutableList.removeFirst()
иMutableList.removeLast()
вkotlin-stdlib
Тип
List
в Java сопоставляется с типомMutableList
в Kotlin. Поскольку API-интерфейсыList.removeFirst()
иList.removeLast()
были представлены в Android 15 (уровень API 35), компилятор Kotlin обрабатывает вызовы функций, напримерlist.removeFirst()
, статически в новые API-интерфейсыList
а не в функции расширения вkotlin-stdlib
.Если приложение перекомпилируется с
compileSdk
равным35
, иminSdk
равным34
или ниже, а затем приложение запускается на Android 14 и более ранних версиях, выдается ошибка времени выполнения:java.lang.NoSuchMethodError: No virtual method removeFirst()Ljava/lang/Object; in class Ljava/util/ArrayList;
Существующая опция
NewApi
lint в плагине Android Gradle может отследить эти новые варианты использования API../gradlew lint
MainActivity.kt:41: Error: Call requires API level 35 (current min is 34): java.util.List#removeFirst [NewApi] list.removeFirst()Чтобы исправить исключения во время выполнения и ошибки проверки, вызовы функций
removeFirst()
иremoveLast()
можно заменить наremoveAt(0)
иremoveAt(list.lastIndex)
соответственно в Kotlin. Если вы используете Android Studio Ladybug | 2024.1.3 или более поздней версии, он также предоставляет возможность быстрого исправления этих ошибок.Рассмотрите возможность удаления
@SuppressLint("NewApi")
иlintOptions { disable 'NewApi' }
если опция lint отключена.Столкновение с другими методами в Java
В существующие типы добавлены новые методы, например
List
иDeque
. Эти новые методы могут быть несовместимы с методами с тем же именем и типами аргументов в других интерфейсах и классах. В случае конфликта сигнатуры метода с несовместимостью компиляторjavac
выводит ошибку времени сборки. Например:Пример ошибки 1:
javac MyList.java
MyList.java:135: error: removeLast() in MyList cannot implement removeLast() in List public void removeLast() { ^ return type void is not compatible with Object where E is a type-variable: E extends Object declared in interface ListПример ошибки 2:
javac MyList.java
MyList.java:7: error: types Deque<Object> and List<Object> are incompatible; public class MyList implements List<Object>, Deque<Object> { both define reversed(), but with unrelated return types 1 errorПример ошибки 3:
javac MyList.java
MyList.java:43: error: types List<E#1> and MyInterface<E#2> are incompatible; public static class MyList implements List<Object>, MyInterface<Object> { class MyList inherits unrelated defaults for getFirst() from types List and MyInterface where E#1,E#2 are type-variables: E#1 extends Object declared in interface List E#2 extends Object declared in interface MyInterface 1 errorЧтобы исправить эти ошибки сборки, класс, реализующий эти интерфейсы, должен переопределить метод совместимым типом возвращаемого значения. Например:
@Override public Object getFirst() { return List.super.getFirst(); }
Безопасность
Android 15 включает изменения, повышающие безопасность системы и помогающие защитить приложения и пользователей от вредоносных приложений.
Запуск защищенной фоновой активности
Android 15 可保护用户免受恶意应用的侵害,并让用户更好地控制 来防止恶意后台应用 将其他应用置于前台、提升其权限以及滥用 用户互动自以下时间以来,后台活动启动一直受到限制 Android 10(API 级别 29)。
禁止与堆栈中的顶部 UID 不匹配的应用启动 activity
恶意应用可以在同一任务中启动另一个应用的 activity,然后
叠加在上面,营造出像该应用一样的错觉。这个“任务”
劫持"攻击绕过了当前的后台启动限制,
会发生在同一个可见任务中。为了降低这种风险,Android 15 新增了
用于阻止与堆栈中的顶层 UID 不匹配的应用启动的标志
活动。如需选择启用应用的所有活动,请更新
allowCrossUidActivitySwitchFromBelow
属性:AndroidManifest.xml
<application android:allowCrossUidActivitySwitchFromBelow="false" >
如果满足以下所有条件,则启用新的安全措施:
- 执行启动的应用以 Android 15 为目标平台。
- 任务堆栈顶部的应用以 Android 15 为目标平台。
- 所有可见活动都已选择启用新保护措施
如果启用了安全措施,应用可能会返回主屏幕,而不是返回 最后一个可见应用(如果他们自行完成任务)。
其他变更
除了限制 UID 匹配之外,这些其他变更也 包括:
- 更改
PendingIntent
创作者,以阻止后台活动启动,具体方法是: 默认。这有助于防止应用意外创建 可能被恶意操作者滥用的PendingIntent
。 - 请勿将应用调到前台,除非
PendingIntent
发送者 允许它。此变更旨在防止恶意应用滥用 在后台启动 activity 的功能。默认情况下,应用 允许将任务堆栈转到前台,除非创建者允许 后台活动启动权限或发送者有后台活动 启动权限 - 控制任务堆栈的顶层 activity 完成其任务的方式。如果 顶层 activity 完成一项任务后,Android 会返回到之前执行的 上次活动时间。此外,如果非顶层 activity 完成其任务,Android 将 返回主屏幕;因此不会阻碍这个非顶层的 活动。
- 防止将其他应用中的任意 activity 启动到您自己的 activity 任务。这项变更旨在防止恶意应用 看起来像是来自其他应用的活动
- 禁止将不可见窗口视为后台活动 发布。这有助于防止恶意应用滥用后台 activity 来向用户显示不需要或恶意的内容。
Более безопасные намерения
В Android 15 представлены новые дополнительные меры безопасности, которые делают действия более безопасными и надежными. Эти изменения направлены на предотвращение потенциальных уязвимостей и неправомерного использования намерений, которые могут быть использованы вредоносными приложениями. В Android 15 есть два основных улучшения безопасности намерений:
- Соответствие целевым фильтрам намерений: намерения, нацеленные на определенные компоненты, должны точно соответствовать спецификациям целевого фильтра намерений. Если вы отправляете намерение запустить действие другого приложения, целевой компонент намерения должен согласовываться с объявленными фильтрами намерений принимающего действия.
- Намерения должны иметь действия: намерения без действия больше не будут соответствовать никаким фильтрам намерений. Это означает, что намерения, используемые для запуска действий или служб, должны иметь четко определенное действие.
Чтобы проверить, как ваше приложение реагирует на эти изменения, используйте StrictMode
в своем приложении. Чтобы просмотреть подробные журналы о нарушениях использования Intent
, добавьте следующий метод:
Котлин
fun onCreate() { StrictMode.setVmPolicy(VmPolicy.Builder() .detectUnsafeIntentLaunch() .build() ) }
Ява
public void onCreate() { StrictMode.setVmPolicy(new VmPolicy.Builder() .detectUnsafeIntentLaunch() .build()); }
Пользовательский опыт и системный интерфейс
Android 15 включает в себя некоторые изменения, призванные создать более последовательный и интуитивно понятный пользовательский интерфейс.
Изменения вставки окна
Есть два изменения, связанных с вставками окон в Android 15: по умолчанию применяется сквозное изображение, а также есть изменения конфигурации, такие как конфигурация системных панелей по умолчанию.
Комплексное обеспечение соблюдения требований
Приложения по умолчанию являются безграничными на устройствах под управлением Android 15, если приложение ориентировано на Android 15 (уровень API 35).
Это критическое изменение, которое может негативно повлиять на пользовательский интерфейс вашего приложения. Изменения затрагивают следующие области пользовательского интерфейса:
- Панель навигации с помощью жестов
- Прозрачный по умолчанию.
- Смещение по нижнему краю отключено, поэтому содержимое отображается за системной панелью навигации, если не применяются вставки.
-
setNavigationBarColor
иR.attr#navigationBarColor
устарели и не влияют на навигацию с помощью жестов. -
setNavigationBarContrastEnforced
иR.attr#navigationBarContrastEnforced
по-прежнему не влияют на навигацию с помощью жестов.
- 3-кнопочная навигация
- По умолчанию непрозрачность установлена на 80%, при этом цвет может совпадать с фоном окна.
- Смещение по нижнему краю отключено, поэтому содержимое отображается за системной панелью навигации, если не применяются вставки.
-
setNavigationBarColor
иR.attr#navigationBarColor
по умолчанию настроены так, чтобы соответствовать фону окна. Чтобы применить это значение по умолчанию, фон окна должен быть окрашен в цвет. Этот API устарел, но продолжает влиять на навигацию с помощью трех кнопок. -
setNavigationBarContrastEnforced
иR.attr#navigationBarContrastEnforced
по умолчанию имеют значение true, что добавляет непрозрачный фон на 80 % при трехкнопочной навигации.
- Строка состояния
- Прозрачный по умолчанию.
- Смещение сверху отключено, поэтому содержимое отображается за строкой состояния, если не применяются вставки.
-
setStatusBarColor
иR.attr#statusBarColor
устарели и не влияют на Android 15. -
setStatusBarContrastEnforced
иR.attr#statusBarContrastEnforced
устарели, но все еще влияют на Android 15.
- Вырез дисплея
-
layoutInDisplayCutoutMode
неплавающих окон должен иметь значениеLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
.SHORT_EDGES
,NEVER
иDEFAULT
интерпретируются какALWAYS
поэтому пользователи не видят черную полосу, вызванную вырезом дисплея, и появляются от края до края.
-
В следующем примере показано приложение до и после ориентации на Android 15 (уровень API 35), а также до и после применения вставок.
Что проверить, если ваше приложение уже является сквозным
Если ваше приложение уже является сквозным и применяет вставки, это в основном не повлияет на вас, за исключением следующих сценариев. Однако даже если вы думаете, что на вас это не повлияет, мы рекомендуем вам протестировать свое приложение.
- У вас есть неплавающее окно, например
Activity
, которое используетSHORT_EDGES
,NEVER
илиDEFAULT
вместоLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
. Если ваше приложение вылетает при запуске, это может быть связано с заставкой. Вы можете либо обновить базовую зависимость заставки до 1.2.0-alpha01 или более поздней, либо установитьwindow.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always
. - Могут быть экраны с меньшим трафиком и закрытым пользовательским интерфейсом. Убедитесь, что на этих менее посещаемых экранах пользовательский интерфейс не перекрыт. Экраны с низким трафиком включают в себя:
- Экраны регистрации или входа в систему
- Страницы настроек
Что проверить, если ваше приложение еще не является сквозным
Если ваше приложение еще не является сквозным, скорее всего, это повлияет на вас. В дополнение к сценариям для приложений, которые уже являются сквозными, следует учитывать следующее:
- Если ваше приложение использует компоненты Material 3 (
androidx.compose.material3
) в компоновке, такие какTopAppBar
,BottomAppBar
иNavigationBar
, эти компоненты, скорее всего, не пострадают, поскольку они автоматически обрабатывают вставки. - Если ваше приложение использует компоненты Material 2 (
androidx.compose.material
) в Compose, эти компоненты не обрабатывают вставки автоматически. Однако вы можете получить доступ к вставкам и применить их вручную. В androidx.compose.material 1.6.0 и более поздних версиях используйте параметрwindowInsets
, чтобы вручную применить вставки дляBottomAppBar
,TopAppBar
,BottomNavigation
иNavigationRail
. Аналогичным образом используйте параметрcontentWindowInsets
дляScaffold
. - Если ваше приложение использует представления и компоненты материалов (
com.google.android.material
), большинство компонентов материалов на основе представлений, таких какBottomNavigationView
,BottomAppBar
,NavigationRailView
илиNavigationView
, обрабатывают вставки и не требуют дополнительной работы. Однако вам необходимо добавитьandroid:fitsSystemWindows="true"
при использованииAppBarLayout
. - Для пользовательских составных элементов примените вставки вручную в качестве заполнения. Если ваш контент находится внутри
Scaffold
, вы можете использовать вставки, используя значения заполненияScaffold
. В противном случае примените дополнение, используя один изWindowInsets
. - Если ваше приложение использует представления и
BottomSheet
,SideSheet
или пользовательские контейнеры, примените заполнение с помощьюViewCompat.setOnApplyWindowInsetsListener
. ДляRecyclerView
примените заполнение с помощью этого прослушивателя, а также добавьтеclipToPadding="false"
.
Что проверить, должно ли ваше приложение предлагать специальную фоновую защиту
Если ваше приложение должно предлагать настраиваемую фоновую защиту для трехкнопочной навигации или строки состояния, ваше приложение должно разместить компонуемый объект или представление за системной панелью с помощью WindowInsets.Type#tappableElement()
чтобы получить высоту трехкнопочной панели навигации или WindowInsets.Type#statusBars
.
Дополнительные сквозные ресурсы
Дополнительные сведения о применении вставок см. в руководствах « Просмотры от края до края» и «Композиция от края до края» .
Устаревшие API
Следующие API устарели, но не отключены:
-
R.attr#enforceStatusBarContrast
-
R.attr#navigationBarColor
(для навигации по 3 кнопкам, с альфа-каналом 80%) -
Window#isStatusBarContrastEnforced
-
Window#setNavigationBarColor
(для навигации по 3 кнопкам, с альфа-значением 80%) -
Window#setStatusBarContrastEnforced
Следующие API устарели и отключены:
-
R.attr#navigationBarColor
(для навигации с помощью жестов) -
R.attr#navigationBarDividerColor
-
R.attr#statusBarColor
-
Window#setDecorFitsSystemWindows
-
Window#getNavigationBarColor
-
Window#getNavigationBarDividerColor
-
Window#getStatusBarColor
-
Window#setNavigationBarColor
(для навигации с помощью жестов) -
Window#setNavigationBarDividerColor
-
Window#setStatusBarColor
Стабильная конфигурация
Если ваше приложение предназначено для Android 15 (уровень API 35) или выше, Configuration
больше не исключает системные панели. Если вы используете размер экрана в классе Configuration
для расчета макета, вам следует заменить его лучшими альтернативами, такими как соответствующие ViewGroup
, WindowInsets
или WindowMetricsCalculator
в зависимости от ваших потребностей.
Configuration
доступна начиная с API 1. Обычно ее получают из Activity.onConfigurationChanged
. Он предоставляет такую информацию, как плотность, ориентация и размеры окон. Одной из важных характеристик размеров окон, возвращаемых из Configuration
является то, что ранее из нее исключались системные панели.
Размер конфигурации обычно используется для выбора ресурсов, например /res/layout-h500dp
, и это по-прежнему допустимый вариант использования. Однако использовать его для расчета планировки всегда не поощрялось. Если вы это сделаете, вам следует отойти от этого сейчас. Вам следует заменить использование Configuration
чем-то более подходящим в зависимости от вашего варианта использования.
Если вы используете его для расчета макета, используйте соответствующую ViewGroup
, такую как CoordinatorLayout
или ConstraintLayout
. Если вы используете его для определения высоты системной панели навигации, используйте WindowInsets
. Если вы хотите узнать текущий размер окна вашего приложения, используйте computeCurrentWindowMetrics
.
В следующем списке описаны поля, затронутые этим изменением:
- Размеры
Configuration.screenWidthDp
иscreenHeightDp
больше не исключают системные панели. -
Configuration.smallestScreenWidthDp
косвенно влияют изменения вscreenWidthDp
иscreenHeightDp
. -
Configuration.orientation
косвенно влияют измененияscreenWidthDp
иscreenHeightDp
на устройствах, близких к квадратным. -
Display.getSize(Point)
косвенно влияют измененияConfiguration
. Это устарело, начиная с уровня API 30. -
Display.getMetrics()
уже работал таким образом, начиная с уровня API 33.
Атрибут ElegantTextHeight по умолчанию имеет значение true
对于以 Android 15(API 级别 35)为目标平台的应用,elegantTextHeight
TextView
属性默认会变为 true
,将默认使用的紧凑字体替换为一些具有较大垂直测量的脚本,使其更易于阅读。紧凑字体旨在防止布局中断;Android 13(API 级别 33)允许文本布局利用 fallbackLineSpacing
属性拉伸垂直高度,从而防止许多此类中断。
在 Android 15 中,系统中仍保留了紧凑字体,因此您的应用可以将 elegantTextHeight
设置为 false
以获得与之前相同的行为,但即将发布的版本不太可能支持此字体。因此,如果您的应用支持以下脚本:阿拉伯语、老挝语、缅甸语、泰米尔语、古吉拉特语、卡纳达语、马拉雅拉姆语、奥里亚语、泰卢固语或泰语,请将 elegantTextHeight
设置为 true
以测试您的应用。
Изменение ширины TextView для сложных форм букв
В предыдущих версиях Android некоторые рукописные шрифты или языки сложной формы могли отображать буквы в области предыдущего или следующего символа. В некоторых случаях такие буквы обрезались в начальной или конечной позиции. Начиная с Android 15, TextView
выделяет ширину для рисования достаточного места для таких букв и позволяет приложениям запрашивать дополнительные поля слева, чтобы предотвратить обрезку.
Поскольку это изменение влияет на то, как TextView
определяет ширину, TextView
по умолчанию выделяет большую ширину, если приложение предназначено для Android 15 (уровень API 35) или выше. Вы можете включить или отключить это поведение, вызвав API setUseBoundsForWidth
в TextView
.
Поскольку добавление левого отступа может привести к смещению существующих макетов, отступы не добавляются по умолчанию даже для приложений, ориентированных на Android 15 или более позднюю версию. Однако вы можете добавить дополнительные отступы для предотвращения обрезки, вызвав setShiftDrawingOffsetForStartOverhang
.
В следующих примерах показано, как эти изменения могут улучшить макет текста для некоторых шрифтов и языков.
Высота строки по умолчанию для EditText с учетом локали
在较低版本的 Android 中,文本布局会拉伸文本的高度,以满足与当前语言区域匹配的字体的行高。例如,如果内容是日语,由于日语字体的行高略高于拉丁字体,因此文本的高度会略高。不过,尽管行高存在这些差异,但无论使用的是哪种语言区域,EditText
元素的大小都是统一的,如下图所示:
对于以 Android 15(API 级别 35)为目标平台的应用,现在为 EditText
预留了最小行高,以匹配指定语言区域的参考字体,如下图所示:
如有需要,您的应用可以将 useLocalePreferredLineHeightForMinimum
属性指定为 false
,以恢复之前的行为;您的应用还可以在 Kotlin 和 Java 中使用 setMinimumFontMetrics
API 设置自定义最小垂直指标。
Камера и медиа
В Android 15 внесены следующие изменения в поведение камеры и мультимедиа для приложений, предназначенных для Android 15 или более поздних версий.
Ограничения на запрос аудиофокуса
Приложения, предназначенные для Android 15 (уровень API 35), должны быть главным приложением или запускать службу переднего плана, чтобы запросить фокус звука . Если приложение пытается запросить фокус, хотя оно не соответствует одному из этих требований, вызов возвращает AUDIOFOCUS_REQUEST_FAILED
.
Дополнительную информацию о фокусе звука можно узнать в разделе «Управление фокусом звука» .
Обновлены ограничения, не связанные с SDK.
Android 15 включает обновленные списки ограниченных интерфейсов, не входящих в SDK, на основе сотрудничества с разработчиками Android и последних результатов внутреннего тестирования. По возможности мы обеспечиваем доступность общедоступных альтернатив, прежде чем ограничивать интерфейсы, не относящиеся к SDK.
Если ваше приложение не предназначено для Android 15, некоторые из этих изменений могут не затронуть вас сразу. Однако, хотя ваше приложение может получить доступ к некоторым интерфейсам, не входящим в SDK, в зависимости от целевого уровня API вашего приложения , использование любого метода или поля, не входящего в SDK, всегда сопряжено с высоким риском поломки вашего приложения.
Если вы не уверены, использует ли ваше приложение интерфейсы, отличные от SDK, вы можете проверить свое приложение, чтобы выяснить это. Если ваше приложение использует интерфейсы, отличные от SDK, вам следует начать планировать переход на альтернативы SDK. Тем не менее, мы понимаем, что в некоторых приложениях есть допустимые варианты использования интерфейсов, отличных от SDK. Если вы не можете найти альтернативу использованию интерфейса, отличного от SDK, для функции вашего приложения, вам следует запросить новый общедоступный API .
Дополнительные сведения об изменениях в этой версии Android см. в разделе Обновления ограничений интерфейса, не связанных с SDK, в Android 15 . Дополнительные сведения об интерфейсах, отличных от SDK, см. в разделе Ограничения на интерфейсы, не относящиеся к SDK .
,Как и предыдущие выпуски, Android 15 включает изменения в поведении, которые могут повлиять на ваше приложение. Следующие изменения поведения применяются исключительно к приложениям, предназначенным для Android 15 или более поздней версии. Если ваше приложение ориентировано на Android 15 или более поздней версии, вам следует изменить его так, чтобы оно правильно поддерживало такое поведение, если это применимо.
Обязательно ознакомьтесь со списком изменений поведения, которые влияют на все приложения, работающие на Android 15, независимо от targetSdkVersion
вашего приложения.
Основная функциональность
Android 15 изменяет или расширяет различные основные возможности системы Android.
Изменения в службах переднего плана
Мы вносим следующие изменения в службы переднего плана в Android 15.
- Поведение времени ожидания службы синхронизации данных на переднем плане
- Новый тип приоритетной службы обработки мультимедиа
- Ограничения на широковещательные приемники
BOOT_COMPLETED
, запускающие службы приоритета - Ограничения на запуск служб переднего плана, пока приложение имеет разрешение
SYSTEM_ALERT_WINDOW
Поведение тайм-аута службы синхронизации данных на переднем плане
В Android 15 представлен новый режим тайм-аута для dataSync
для приложений, предназначенных для Android 15 (уровень API 35) или выше. Это поведение также применимо к новому типу службы переднего плана mediaProcessing
.
Система разрешает службам dataSync
приложения работать в общей сложности 6 часов в течение 24-часового периода, после чего система вызывает метод Service.onTimeout(int, int)
работающей службы (представленный в Android 15). В это время у службы есть несколько секунд для вызова Service.stopSelf()
. При вызове Service.onTimeout()
служба больше не считается службой переднего плана. Если служба не вызывает Service.stopSelf()
, система выдает внутреннее исключение. Исключение регистрируется в Logcat со следующим сообщением:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type dataSync did not stop within its timeout: [component name]"
Чтобы избежать проблем с этим изменением поведения, вы можете выполнить одно или несколько из следующих действий:
- Пусть ваш сервис реализует новый метод
Service.onTimeout(int, int)
. Когда ваше приложение получит обратный вызов, обязательно вызовитеstopSelf()
в течение нескольких секунд. (Если вы не остановите приложение сразу, система выдаст ошибку.) - Убедитесь, что службы
dataSync
вашего приложения не работают более 6 часов в течение 24-часового периода (если только пользователь не взаимодействует с приложением, сбрасывая таймер). - Запускайте службы
dataSync
переднего плана только в результате прямого взаимодействия с пользователем; поскольку ваше приложение находится на переднем плане при запуске службы, у вашей службы есть полные шесть часов после того, как приложение перейдет в фоновый режим. - Вместо использования службы переднего плана
dataSync
используйте альтернативный API .
Если службы приоритетного плана dataSync
вашего приложения работали в течение 6 часов за последние 24 часа, вы не сможете запустить другую службу приоритетного плана dataSync
, пока пользователь не переведет ваше приложение на передний план (что сбрасывает таймер). Если вы попытаетесь запустить другую службу переднего плана dataSync
, система выдаст исключение ForegroundServiceStartNotAllowedException
с сообщением об ошибке, например «Ограничение времени для типа службы переднего плана dataSync уже исчерпано».
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить таймауты синхронизации данных, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить таймауты, выполните следующую команду adb
:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
Вы также можете настроить период ожидания, чтобы было проще протестировать поведение вашего приложения при достижении предела. Чтобы установить новый период ожидания, выполните следующую команду adb
:
adb shell device_config put activity_manager data_sync_fgs_timeout_duration duration-in-milliseconds
Новый тип приоритетной службы обработки мультимедиа
В Android 15 представлен новый тип службы переднего плана — mediaProcessing
. Этот тип службы подходит для таких операций, как перекодирование медиафайлов. Например, мультимедийное приложение может загрузить аудиофайл, и ему необходимо преобразовать его в другой формат перед воспроизведением. Вы можете использовать службу переднего плана mediaProcessing
чтобы гарантировать, что преобразование продолжается, даже когда приложение находится в фоновом режиме.
Система разрешает службам mediaProcessing
приложения работать в общей сложности 6 часов в течение 24-часового периода, после чего система вызывает метод Service.onTimeout(int, int)
работающей службы (представленный в Android 15). В это время у службы есть несколько секунд для вызова Service.stopSelf()
. Если служба не вызывает Service.stopSelf()
, система выдает внутреннее исключение. Исключение регистрируется в Logcat со следующим сообщением:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type mediaProcessing did not stop within its timeout: [component name]"
Чтобы избежать исключения, вы можете сделать одно из следующих действий:
- Пусть ваш сервис реализует новый метод
Service.onTimeout(int, int)
. Когда ваше приложение получит обратный вызов, обязательно вызовитеstopSelf()
в течение нескольких секунд. (Если вы не остановите приложение сразу, система выдаст ошибку.) - Убедитесь, что службы
mediaProcessing
вашего приложения не работают более 6 часов в течение любого 24-часового периода (если только пользователь не взаимодействует с приложением, сбрасывая таймер). - Запускайте службы переднего плана
mediaProcessing
только в результате прямого взаимодействия с пользователем; поскольку ваше приложение находится на переднем плане при запуске службы, у вашей службы есть полные шесть часов после того, как приложение перейдет в фоновый режим. - Вместо использования службы переднего плана
mediaProcessing
используйте альтернативный API , например WorkManager.
Если службы переднего плана mediaProcessing
вашего приложения работали в течение 6 часов за последние 24, вы не сможете запустить другую службу переднего плана mediaProcessing
, пока пользователь не переведет ваше приложение на передний план (что сбрасывает таймер). Если вы попытаетесь запустить другую службу переднего плана mediaProcessing
, система выдаст ForegroundServiceStartNotAllowedException
с сообщением об ошибке, например «Ограничение времени уже исчерпано для типа службы переднего плана mediaProcessing».
Дополнительные сведения о типе службы mediaProcessing
см. в разделе Изменения в типах служб переднего плана для Android 15: обработка мультимедиа .
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить тайм-ауты обработки мультимедиа, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить таймауты, выполните следующую команду adb
:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
Вы также можете настроить период ожидания, чтобы было проще протестировать поведение вашего приложения при достижении предела. Чтобы установить новый период ожидания, выполните следующую команду adb
:
adb shell device_config put activity_manager media_processing_fgs_timeout_duration duration-in-milliseconds
Ограничения на широковещательные приемники BOOT_COMPLETED
, запускающие службы приоритета
在启动 BOOT_COMPLETED
广播接收器方面存在新限制
前台服务。BOOT_COMPLETED
接收器不能启动
以下类型的前台服务:
dataSync
camera
mediaPlayback
phoneCall
mediaProjection
microphone
(自 Android 14 起,microphone
就受到此限制)
如果 BOOT_COMPLETED
接收器尝试启动任何上述类型的前台
服务,系统会抛出 ForegroundServiceStartNotAllowedException
。
测试
如需测试应用的行为,您可以启用这些新限制,即使您的应用并未以 Android 15 为目标平台(只要应用在 Android 15 设备上运行)也是如此。运行以下 adb
命令:
adb shell am compat enable FGS_BOOT_COMPLETED_RESTRICTIONS your-package-name
如需在不重启设备的情况下发送 BOOT_COMPLETED
广播,请运行以下 adb
命令:
adb shell am broadcast -a android.intent.action.BOOT_COMPLETED your-package-name
Ограничения на запуск служб переднего плана, пока приложение имеет разрешение SYSTEM_ALERT_WINDOW
Раньше, если приложение имело разрешение SYSTEM_ALERT_WINDOW
, оно могло запустить службу переднего плана, даже если приложение в данный момент находилось в фоновом режиме (как обсуждалось в разделе об исключениях из ограничений фонового запуска ).
Если приложение предназначено для Android 15, это исключение теперь уже. Теперь приложению необходимо иметь разрешение SYSTEM_ALERT_WINDOW
, а также иметь видимое окно наложения. То есть приложению необходимо сначала запустить окно TYPE_APPLICATION_OVERLAY
, и это окно должно быть видимым перед запуском службы переднего плана.
Если ваше приложение пытается запустить службу переднего плана в фоновом режиме, не отвечая этим новым требованиям (и у него нет других исключений), система выдает ForegroundServiceStartNotAllowedException
.
Если ваше приложение объявляет разрешение SYSTEM_ALERT_WINDOW
и запускает службы переднего плана в фоновом режиме, это изменение может повлиять на него. Если ваше приложение получает исключение ForegroundServiceStartNotAllowedException
, проверьте порядок операций вашего приложения и убедитесь, что в нем уже есть активное окно наложения, прежде чем оно попытается запустить службу переднего плана из фона. Вы можете проверить, видимо ли ваше окно наложения в данный момент, вызвав View.getWindowVisibility()
, или вы можете переопределить View.onWindowVisibilityChanged()
чтобы получать уведомления при каждом изменении видимости.
Тестирование
Чтобы проверить поведение вашего приложения, вы можете включить эти новые ограничения, даже если ваше приложение не предназначено для Android 15 (при условии, что приложение работает на устройстве Android 15). Чтобы включить эти новые ограничения на запуск служб переднего плана в фоновом режиме, выполните следующую команду adb
:
adb shell am compat enable FGS_SAW_RESTRICTIONS your-package-name
Изменения в том, когда приложения могут изменять глобальное состояние режима «Не беспокоить».
Приложения, предназначенные для Android 15 (уровень API 35) и выше, больше не могут изменять глобальное состояние или политику режима «Не беспокоить» (DND) на устройстве (путем изменения пользовательских настроек или отключения режима «Не беспокоить»). Вместо этого приложения должны внести AutomaticZenRule
, которое система объединяет в глобальную политику с существующей схемой наиболее ограничительной политики. Вызовы к существующим API, которые ранее влияли на глобальное состояние ( setInterruptionFilter
, setNotificationPolicy
), приводят к созданию или обновлению неявного AutomaticZenRule
, который включается и выключается в зависимости от цикла вызовов этих вызовов API.
Обратите внимание, что это изменение влияет на наблюдаемое поведение только в том случае, если приложение вызывает setInterruptionFilter(INTERRUPTION_FILTER_ALL)
и ожидает, что этот вызов деактивирует AutomaticZenRule
, который ранее был активирован его владельцами.
Изменения API OpenJDK
В Android 15 продолжается работа по обновлению основных библиотек Android, чтобы они соответствовали функциям последних выпусков OpenJDK LTS.
Некоторые из этих изменений могут повлиять на совместимость приложений, предназначенных для Android 15 (уровень API 35):
Изменения в API форматирования строк : проверка индекса аргумента, флагов, ширины и точности теперь более строгая при использовании следующих API
String.format()
иFormatter.format()
:-
String.format(String, Object[])
-
String.format(Locale, String, Object[])
-
Formatter.format(String, Object[])
-
Formatter.format(Locale, String, Object[])
Например, следующее исключение выдается, когда используется индекс аргумента 0 (
%0
в строке формата):IllegalFormatArgumentIndexException: Illegal format argument index = 0
В этом случае проблему можно решить, используя индекс аргумента 1 (
%1
в строке формата).-
Изменения в типе компонента
Arrays.asList(...).toArray()
: при использованииArrays.asList(...).toArray()
типом компонента результирующего массива теперь являетсяObject
, а не тип элементы базового массива. Итак, следующий код генерирует исключениеClassCastException
:String[] elements = (String[]) Arrays.asList("one", "two").toArray();
В этом случае, чтобы сохранить
String
как тип компонента в результирующем массиве, вы можете вместо этого использоватьCollection.toArray(Object[])
:String[] elements = Arrays.asList("two", "one").toArray(new String[0]);
Изменения в обработке языковых кодов . При использовании
Locale
API языковые коды для иврита, идиша и индонезийского языка больше не преобразуются в их устаревшие формы (иврит:iw
, идиш:ji
и индонезийский:in
). При указании кода языка для одного из этих языков вместо этого используйте коды из ISO 639-1 (иврит:he
, идиш:yi
и индонезийский:id
).Изменения в случайных последовательностях целых чисел : после изменений, внесенных в https://bugs.openjdk.org/browse/JDK-8301574 , следующие методы
Random.ints()
теперь возвращают последовательность чисел, отличную от методовRandom.nextInt()
. делать:Как правило, это изменение не должно приводить к нарушению работы приложения, но ваш код не должен ожидать, что последовательность, сгенерированная методами
Random.ints()
будет соответствоватьRandom.nextInt()
.
Новый API SequencedCollection
может повлиять на совместимость вашего приложения после обновления compileSdk
в конфигурации сборки вашего приложения для использования Android 15 (уровень API 35) :
Конфликт с функциями расширения
MutableList.removeFirst()
иMutableList.removeLast()
вkotlin-stdlib
Тип
List
в Java сопоставляется с типомMutableList
в Kotlin. Поскольку API-интерфейсыList.removeFirst()
иList.removeLast()
были представлены в Android 15 (уровень API 35), компилятор Kotlin обрабатывает вызовы функций, напримерlist.removeFirst()
, статически в новые API-интерфейсыList
а не в функции расширения вkotlin-stdlib
.Если приложение перекомпилируется с
compileSdk
равным35
, иminSdk
равным34
или ниже, а затем приложение запускается на Android 14 и более ранних версиях, выдается ошибка времени выполнения:java.lang.NoSuchMethodError: No virtual method removeFirst()Ljava/lang/Object; in class Ljava/util/ArrayList;
Существующая опция
NewApi
lint в плагине Android Gradle может отследить эти новые варианты использования API../gradlew lint
MainActivity.kt:41: Error: Call requires API level 35 (current min is 34): java.util.List#removeFirst [NewApi] list.removeFirst()Чтобы исправить исключения во время выполнения и ошибки проверки, вызовы функций
removeFirst()
иremoveLast()
можно заменить наremoveAt(0)
иremoveAt(list.lastIndex)
соответственно в Kotlin. Если вы используете Android Studio Ladybug | 2024.1.3 или более поздней версии, он также предоставляет возможность быстрого исправления этих ошибок.Рассмотрите возможность удаления
@SuppressLint("NewApi")
иlintOptions { disable 'NewApi' }
если опция lint отключена.Столкновение с другими методами в Java
В существующие типы добавлены новые методы, например
List
иDeque
. Эти новые методы могут быть несовместимы с методами с тем же именем и типами аргументов в других интерфейсах и классах. В случае конфликта сигнатуры метода с несовместимостью компиляторjavac
выводит ошибку времени сборки. Например:Пример ошибки 1:
javac MyList.java
MyList.java:135: error: removeLast() in MyList cannot implement removeLast() in List public void removeLast() { ^ return type void is not compatible with Object where E is a type-variable: E extends Object declared in interface ListПример ошибки 2:
javac MyList.java
MyList.java:7: error: types Deque<Object> and List<Object> are incompatible; public class MyList implements List<Object>, Deque<Object> { both define reversed(), but with unrelated return types 1 errorПример ошибки 3:
javac MyList.java
MyList.java:43: error: types List<E#1> and MyInterface<E#2> are incompatible; public static class MyList implements List<Object>, MyInterface<Object> { class MyList inherits unrelated defaults for getFirst() from types List and MyInterface where E#1,E#2 are type-variables: E#1 extends Object declared in interface List E#2 extends Object declared in interface MyInterface 1 errorЧтобы исправить эти ошибки сборки, класс, реализующий эти интерфейсы, должен переопределить метод совместимым типом возвращаемого значения. Например:
@Override public Object getFirst() { return List.super.getFirst(); }
Безопасность
Android 15 включает изменения, повышающие безопасность системы и помогающие защитить приложения и пользователей от вредоносных приложений.
Запуск защищенной фоновой активности
Android 15 可保护用户免受恶意应用的侵害,并让用户更好地控制 来防止恶意后台应用 将其他应用置于前台、提升其权限以及滥用 用户互动自以下时间以来,后台活动启动一直受到限制 Android 10(API 级别 29)。
禁止与堆栈中的顶部 UID 不匹配的应用启动 activity
恶意应用可以在同一任务中启动另一个应用的 activity,然后
叠加在上面,营造出像该应用一样的错觉。这个“任务”
劫持"攻击绕过了当前的后台启动限制,
会发生在同一个可见任务中。为了降低这种风险,Android 15 新增了
用于阻止与堆栈中的顶层 UID 不匹配的应用启动的标志
活动。如需选择启用应用的所有活动,请更新
allowCrossUidActivitySwitchFromBelow
属性:AndroidManifest.xml
<application android:allowCrossUidActivitySwitchFromBelow="false" >
如果满足以下所有条件,则启用新的安全措施:
- 执行启动的应用以 Android 15 为目标平台。
- 任务堆栈顶部的应用以 Android 15 为目标平台。
- 所有可见活动都已选择启用新保护措施
如果启用了安全措施,应用可能会返回主屏幕,而不是返回 最后一个可见应用(如果他们自行完成任务)。
其他变更
除了限制 UID 匹配之外,这些其他变更也 包括:
- 更改
PendingIntent
创作者,以阻止后台活动启动,具体方法是: 默认。这有助于防止应用意外创建 可能被恶意操作者滥用的PendingIntent
。 - 请勿将应用调到前台,除非
PendingIntent
发送者 允许它。此变更旨在防止恶意应用滥用 在后台启动 activity 的功能。默认情况下,应用 允许将任务堆栈转到前台,除非创建者允许 后台活动启动权限或发送者有后台活动 启动权限 - 控制任务堆栈的顶层 activity 完成其任务的方式。如果 顶层 activity 完成一项任务后,Android 会返回到之前执行的 上次活动时间。此外,如果非顶层 activity 完成其任务,Android 将 返回主屏幕;因此不会阻碍这个非顶层的 活动。
- 防止将其他应用中的任意 activity 启动到您自己的 activity 任务。这项变更旨在防止恶意应用 看起来像是来自其他应用的活动
- 禁止将不可见窗口视为后台活动 发布。这有助于防止恶意应用滥用后台 activity 来向用户显示不需要或恶意的内容。
Более безопасные намерения
В Android 15 представлены новые дополнительные меры безопасности, которые делают действия более безопасными и надежными. Эти изменения направлены на предотвращение потенциальных уязвимостей и неправомерного использования намерений, которые могут быть использованы вредоносными приложениями. В Android 15 есть два основных улучшения безопасности намерений:
- Соответствие целевым фильтрам намерений: намерения, нацеленные на определенные компоненты, должны точно соответствовать спецификациям целевого фильтра намерений. Если вы отправляете намерение запустить действие другого приложения, целевой компонент намерения должен согласовываться с объявленными фильтрами намерений принимающего действия.
- Намерения должны иметь действия: намерения без действия больше не будут соответствовать никаким фильтрам намерений. Это означает, что намерения, используемые для запуска действий или служб, должны иметь четко определенное действие.
Чтобы проверить, как ваше приложение реагирует на эти изменения, используйте StrictMode
в своем приложении. Чтобы просмотреть подробные журналы о нарушениях использования Intent
, добавьте следующий метод:
Котлин
fun onCreate() { StrictMode.setVmPolicy(VmPolicy.Builder() .detectUnsafeIntentLaunch() .build() ) }
Ява
public void onCreate() { StrictMode.setVmPolicy(new VmPolicy.Builder() .detectUnsafeIntentLaunch() .build()); }
Пользовательский опыт и системный интерфейс
Android 15 включает в себя некоторые изменения, призванные создать более последовательный и интуитивно понятный пользовательский интерфейс.
Изменения вставки окна
Есть два изменения, связанных с вставками окон в Android 15: по умолчанию применяется сквозное изображение, а также есть изменения конфигурации, такие как конфигурация системных панелей по умолчанию.
Комплексное обеспечение соблюдения требований
Приложения по умолчанию являются безграничными на устройствах под управлением Android 15, если приложение ориентировано на Android 15 (уровень API 35).
Это критическое изменение, которое может негативно повлиять на пользовательский интерфейс вашего приложения. Изменения затрагивают следующие области пользовательского интерфейса:
- Панель навигации с помощью жестов
- Прозрачный по умолчанию.
- Смещение по нижнему краю отключено, поэтому содержимое отображается за системной панелью навигации, если не применяются вставки.
-
setNavigationBarColor
иR.attr#navigationBarColor
устарели и не влияют на навигацию с помощью жестов. -
setNavigationBarContrastEnforced
иR.attr#navigationBarContrastEnforced
по-прежнему не влияют на навигацию с помощью жестов.
- 3-кнопочная навигация
- По умолчанию непрозрачность установлена на 80%, при этом цвет может совпадать с фоном окна.
- Смещение по нижнему краю отключено, поэтому содержимое отображается за системной панелью навигации, если не применяются вставки.
-
setNavigationBarColor
иR.attr#navigationBarColor
по умолчанию настроены так, чтобы соответствовать фону окна. Чтобы применить это значение по умолчанию, фон окна должен быть окрашен в цвет. Этот API устарел, но продолжает влиять на навигацию с помощью трех кнопок. -
setNavigationBarContrastEnforced
иR.attr#navigationBarContrastEnforced
по умолчанию имеют значение true, что добавляет непрозрачный фон на 80 % при трехкнопочной навигации.
- Строка состояния
- Прозрачный по умолчанию.
- Смещение сверху отключено, поэтому содержимое отображается за строкой состояния, если не применяются вставки.
-
setStatusBarColor
иR.attr#statusBarColor
устарели и не влияют на Android 15. -
setStatusBarContrastEnforced
иR.attr#statusBarContrastEnforced
устарели, но все еще влияют на Android 15.
- Вырез дисплея
-
layoutInDisplayCutoutMode
неплавающих окон должен иметь значениеLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
.SHORT_EDGES
,NEVER
иDEFAULT
интерпретируются какALWAYS
поэтому пользователи не видят черную полосу, вызванную вырезом дисплея, и появляются от края до края.
-
В следующем примере показано приложение до и после ориентации на Android 15 (уровень API 35), а также до и после применения вставок.
Что проверить, если ваше приложение уже является сквозным
Если ваше приложение уже является сквозным и применяет вставки, это в основном не повлияет на вас, за исключением следующих сценариев. Однако даже если вы думаете, что на вас это не повлияет, мы рекомендуем вам протестировать свое приложение.
- У вас есть неплавающее окно, например
Activity
, которое используетSHORT_EDGES
,NEVER
илиDEFAULT
вместоLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
. Если ваше приложение вылетает при запуске, это может быть связано с заставкой. Вы можете либо обновить базовую зависимость заставки до 1.2.0-alpha01 или более поздней, либо установитьwindow.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always
. - Могут быть экраны с меньшим трафиком и закрытым пользовательским интерфейсом. Убедитесь, что на этих менее посещаемых экранах пользовательский интерфейс не перекрыт. Экраны с низким трафиком включают в себя:
- Экраны регистрации или входа в систему
- Страницы настроек
Что проверить, если ваше приложение еще не является сквозным
Если ваше приложение еще не является сквозным, скорее всего, это повлияет на вас. В дополнение к сценариям для приложений, которые уже являются сквозными, следует учитывать следующее:
- Если ваше приложение использует компоненты Material 3 (
androidx.compose.material3
) в компоновке, такие какTopAppBar
,BottomAppBar
иNavigationBar
, эти компоненты, скорее всего, не пострадают, поскольку они автоматически обрабатывают вставки. - Если ваше приложение использует компоненты Material 2 (
androidx.compose.material
) в Compose, эти компоненты не обрабатывают вставки автоматически. Однако вы можете получить доступ к вставкам и применить их вручную. В androidx.compose.material 1.6.0 и более поздних версиях используйте параметрwindowInsets
, чтобы вручную применить вставки дляBottomAppBar
,TopAppBar
,BottomNavigation
иNavigationRail
. Аналогичным образом используйте параметрcontentWindowInsets
дляScaffold
. - Если ваше приложение использует представления и компоненты материалов (
com.google.android.material
), большинство компонентов материалов на основе представлений, таких какBottomNavigationView
,BottomAppBar
,NavigationRailView
илиNavigationView
, обрабатывают вставки и не требуют дополнительной работы. Однако вам необходимо добавитьandroid:fitsSystemWindows="true"
при использованииAppBarLayout
. - Для пользовательских составных элементов примените вставки вручную в качестве заполнения. Если ваш контент находится внутри
Scaffold
, вы можете использовать вставки, используя значения заполненияScaffold
. В противном случае примените дополнение, используя один изWindowInsets
. - Если ваше приложение использует представления и
BottomSheet
,SideSheet
или пользовательские контейнеры, примените заполнение с помощьюViewCompat.setOnApplyWindowInsetsListener
. ДляRecyclerView
примените заполнение с помощью этого прослушивателя, а также добавьтеclipToPadding="false"
.
Что проверить, должно ли ваше приложение предлагать специальную фоновую защиту
Если ваше приложение должно предлагать настраиваемую фоновую защиту для трехкнопочной навигации или строки состояния, ваше приложение должно разместить компонуемый объект или представление за системной панелью с помощью WindowInsets.Type#tappableElement()
чтобы получить высоту трехкнопочной панели навигации или WindowInsets.Type#statusBars
.
Дополнительные сквозные ресурсы
Дополнительные сведения о применении вставок см. в руководствах « Просмотры от края до края» и «Композиция от края до края» .
Устаревшие API
Следующие API устарели, но не отключены:
-
R.attr#enforceStatusBarContrast
-
R.attr#navigationBarColor
(для навигации по 3 кнопкам, с альфа-каналом 80%) -
Window#isStatusBarContrastEnforced
-
Window#setNavigationBarColor
(для навигации по 3 кнопкам, с альфа-значением 80%) -
Window#setStatusBarContrastEnforced
Следующие API устарели и отключены:
-
R.attr#navigationBarColor
(для навигации с помощью жестов) -
R.attr#navigationBarDividerColor
-
R.attr#statusBarColor
-
Window#setDecorFitsSystemWindows
-
Window#getNavigationBarColor
-
Window#getNavigationBarDividerColor
-
Window#getStatusBarColor
-
Window#setNavigationBarColor
(для навигации с помощью жестов) -
Window#setNavigationBarDividerColor
-
Window#setStatusBarColor
Стабильная конфигурация
Если ваше приложение предназначено для Android 15 (уровень API 35) или выше, Configuration
больше не исключает системные панели. Если вы используете размер экрана в классе Configuration
для расчета макета, вам следует заменить его лучшими альтернативами, такими как соответствующие ViewGroup
, WindowInsets
или WindowMetricsCalculator
в зависимости от ваших потребностей.
Configuration
доступна начиная с API 1. Обычно ее получают из Activity.onConfigurationChanged
. Он предоставляет такую информацию, как плотность, ориентация и размеры окон. Одной из важных характеристик размеров окон, возвращаемых из Configuration
является то, что ранее из нее исключались системные панели.
Размер конфигурации обычно используется для выбора ресурсов, например /res/layout-h500dp
, и это по-прежнему допустимый вариант использования. Однако использовать его для расчета планировки всегда не поощрялось. Если вы это сделаете, вам следует отойти от этого сейчас. Вам следует заменить использование Configuration
чем-то более подходящим в зависимости от вашего варианта использования.
Если вы используете его для расчета макета, используйте соответствующую ViewGroup
, такую как CoordinatorLayout
или ConstraintLayout
. Если вы используете его для определения высоты системной панели навигации, используйте WindowInsets
. Если вы хотите узнать текущий размер окна вашего приложения, используйте computeCurrentWindowMetrics
.
В следующем списке описаны поля, затронутые этим изменением:
- Размеры
Configuration.screenWidthDp
иscreenHeightDp
больше не исключают системные панели. -
Configuration.smallestScreenWidthDp
косвенно влияют изменения вscreenWidthDp
иscreenHeightDp
. -
Configuration.orientation
косвенно влияют измененияscreenWidthDp
иscreenHeightDp
на устройствах, близких к квадратным. -
Display.getSize(Point)
косвенно влияют измененияConfiguration
. Это устарело, начиная с уровня API 30. -
Display.getMetrics()
уже работал таким образом, начиная с уровня API 33.
Атрибут ElegantTextHeight по умолчанию имеет значение true
对于以 Android 15(API 级别 35)为目标平台的应用,elegantTextHeight
TextView
属性默认会变为 true
,将默认使用的紧凑字体替换为一些具有较大垂直测量的脚本,使其更易于阅读。紧凑字体旨在防止布局中断;Android 13(API 级别 33)允许文本布局利用 fallbackLineSpacing
属性拉伸垂直高度,从而防止许多此类中断。
在 Android 15 中,系统中仍保留了紧凑字体,因此您的应用可以将 elegantTextHeight
设置为 false
以获得与之前相同的行为,但即将发布的版本不太可能支持此字体。因此,如果您的应用支持以下脚本:阿拉伯语、老挝语、缅甸语、泰米尔语、古吉拉特语、卡纳达语、马拉雅拉姆语、奥里亚语、泰卢固语或泰语,请将 elegantTextHeight
设置为 true
以测试您的应用。
Изменение ширины TextView для сложных форм букв
В предыдущих версиях Android некоторые рукописные шрифты или языки сложной формы могли отображать буквы в области предыдущего или следующего символа. В некоторых случаях такие буквы обрезались в начальной или конечной позиции. Начиная с Android 15, TextView
выделяет ширину для рисования достаточного места для таких букв и позволяет приложениям запрашивать дополнительные поля слева, чтобы предотвратить обрезку.
Поскольку это изменение влияет на то, как TextView
определяет ширину, TextView
по умолчанию выделяет большую ширину, если приложение предназначено для Android 15 (уровень API 35) или выше. Вы можете включить или отключить это поведение, вызвав API setUseBoundsForWidth
в TextView
.
Поскольку добавление левого отступа может привести к смещению существующих макетов, отступы не добавляются по умолчанию даже для приложений, ориентированных на Android 15 или более позднюю версию. Однако вы можете добавить дополнительные отступы для предотвращения обрезки, вызвав setShiftDrawingOffsetForStartOverhang
.
В следующих примерах показано, как эти изменения могут улучшить макет текста для некоторых шрифтов и языков.
Высота строки по умолчанию для EditText с учетом локали
在较低版本的 Android 中,文本布局会拉伸文本的高度,以满足与当前语言区域匹配的字体的行高。例如,如果内容是日语,由于日语字体的行高略高于拉丁字体,因此文本的高度会略高。不过,尽管行高存在这些差异,但无论使用的是哪种语言区域,EditText
元素的大小都是统一的,如下图所示:
对于以 Android 15(API 级别 35)为目标平台的应用,现在为 EditText
预留了最小行高,以匹配指定语言区域的参考字体,如下图所示:
如有需要,您的应用可以将 useLocalePreferredLineHeightForMinimum
属性指定为 false
,以恢复之前的行为;您的应用还可以在 Kotlin 和 Java 中使用 setMinimumFontMetrics
API 设置自定义最小垂直指标。
Камера и медиа
В Android 15 внесены следующие изменения в поведение камеры и мультимедиа для приложений, предназначенных для Android 15 или более поздних версий.
Ограничения на запрос аудиофокуса
Приложения, предназначенные для Android 15 (уровень API 35), должны быть главным приложением или запускать службу переднего плана, чтобы запросить фокус звука . Если приложение пытается запросить фокус, хотя оно не соответствует одному из этих требований, вызов возвращает AUDIOFOCUS_REQUEST_FAILED
.
Дополнительную информацию о фокусе звука можно узнать в разделе «Управление фокусом звука» .
Обновлены ограничения, не связанные с SDK.
Android 15 включает обновленные списки ограниченных интерфейсов, не входящих в SDK, на основе сотрудничества с разработчиками Android и последних результатов внутреннего тестирования. По возможности мы обеспечиваем доступность общедоступных альтернатив, прежде чем ограничивать интерфейсы, не относящиеся к SDK.
Если ваше приложение не предназначено для Android 15, некоторые из этих изменений могут не затронуть вас сразу. Однако, хотя ваше приложение может получить доступ к некоторым интерфейсам, не входящим в SDK, в зависимости от целевого уровня API вашего приложения , использование любого метода или поля, не входящего в SDK, всегда сопряжено с высоким риском поломки вашего приложения.
Если вы не уверены, использует ли ваше приложение интерфейсы, отличные от SDK, вы можете проверить свое приложение, чтобы выяснить это. Если ваше приложение использует интерфейсы, отличные от SDK, вам следует начать планировать переход на альтернативы SDK. Тем не менее, мы понимаем, что в некоторых приложениях есть допустимые варианты использования интерфейсов, отличных от SDK. Если вы не можете найти альтернативу использованию интерфейса, отличного от SDK, для функции вашего приложения, вам следует запросить новый общедоступный API .
Дополнительные сведения об изменениях в этой версии Android см. в разделе Обновления ограничений интерфейса, не связанных с SDK, в Android 15 . Дополнительные сведения об интерфейсах, отличных от SDK, см. в разделе Ограничения на интерфейсы, не относящиеся к SDK .