Как и предыдущие выпуски, 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 introduces a new foreground service type, mediaProcessing
. This
service type is appropriate for operations like transcoding media files. For
example, a media app might download an audio file and need to convert it to a
different format before playing it. You can use a mediaProcessing
foreground
service to make sure the conversion continues even while the app is in the
background.
The system permits an app's mediaProcessing
services to run for a total of 6
hours in a 24-hour period, after which the system calls the running service's
Service.onTimeout(int, int)
method (introduced in Android
15). At this time, the service has a few seconds to call
Service.stopSelf()
. If the service does not
call Service.stopSelf()
, the system throws an internal exception. The
exception is logged in Logcat with the following message:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type mediaProcessing did not stop within its timeout: [component name]"
To avoid having the exception, you can do one of the following:
- Have your service implement the new
Service.onTimeout(int, int)
method. When your app receives the callback, make sure to callstopSelf()
within a few seconds. (If you don't stop the app right away, the system generates a failure.) - Make sure your app's
mediaProcessing
services don't run for more than a total of 6 hours in any 24-hour period (unless the user interacts with the app, resetting the timer). - Only start
mediaProcessing
foreground services as a result of direct user interaction; since your app is in the foreground when the service starts, your service has the full six hours after the app goes to the background. - Instead of using a
mediaProcessing
foreground service, use an alternative API, like WorkManager.
If your app's mediaProcessing
foreground services have run for 6 hours in the
last 24, you cannot start another mediaProcessing
foreground service unless
the user has brought your app to the foreground (which resets the timer). If you
try to start another mediaProcessing
foreground service, the system throws
ForegroundServiceStartNotAllowedException
with an error message like "Time limit already exhausted for foreground service
type mediaProcessing".
For more information about the mediaProcessing
service type, see Changes to
foreground service types for Android 15: Media processing.
Testing
To test your app's behavior, you can enable media processing timeouts even if
your app is not targeting Android 15 (as long as the app is running on an
Android 15 device). To enable timeouts, run the following adb
command:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
You can also adjust the timeout period, to make it easier to test how your
app behaves when the limit is reached. To set a new timeout period, run the
following adb
command:
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
(это ограничение действует дляmicrophone
начиная с Android 14)
Если получатель 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 включает изменения, повышающие безопасность системы и помогающие защитить приложения и пользователей от вредоносных приложений.
Ограниченные версии TLS
Android 15 ограничивает использование TLS версий 1.0 и 1.1. Эти версии ранее были признаны устаревшими в Android, но теперь запрещены для приложений, ориентированных на Android 15.
Запуск защищенной фоновой активности
Android 15 защищает пользователей от вредоносных приложений и дает им больше контроля над своими устройствами, добавляя изменения, которые не позволяют вредоносным фоновым приложениям выводить другие приложения на передний план, повышать их привилегии и злоупотреблять взаимодействием с пользователем. Запуск фоновой активности ограничен начиная с Android 10 (уровень API 29).
Блокировать запуск действий приложений, которые не соответствуют верхнему UID в стеке.
Вредоносные приложения могут запускать действия другого приложения в рамках той же задачи, а затем накладываться поверх них, создавая иллюзию того, что они являются этим приложением. Эта атака «перехвата задачи» обходит текущие ограничения на фоновый запуск, поскольку все это происходит в рамках одной и той же видимой задачи. Чтобы снизить этот риск, в Android 15 добавлен флаг, который блокирует запуск действий приложений, которые не соответствуют верхнему UID в стеке. Чтобы разрешить все действия вашего приложения, обновите allowCrossUidActivitySwitchFromBelow
в файле AndroidManifest.xml
вашего приложения:
<application android:allowCrossUidActivitySwitchFromBelow="false" >
Новые меры безопасности активны, если выполняются все следующие условия:
- Приложение, осуществляющее запуск, ориентировано на Android 15.
- Приложение на вершине стека задач предназначено для Android 15.
- Любая видимая активность включает новые средства защиты.
Если меры безопасности включены, приложения могут вернуться домой, а не к последнему видимому приложению, если завершат свою задачу.
Другие изменения
Помимо ограничения на сопоставление UID, включены и другие изменения:
- Измените создателей
PendingIntent
, чтобы они по умолчанию блокировали запуск фоновой активности . Это помогает предотвратить случайное создание приложениямиPendingIntent
, которым могут воспользоваться злоумышленники. - Не выводите приложение на передний план, если это не разрешено отправителем
PendingIntent
. Это изменение направлено на то, чтобы вредоносные приложения не злоупотребляли возможностью запускать действия в фоновом режиме. По умолчанию приложениям не разрешено выводить стек задач на передний план, если только создатель не разрешил права запуска фоновых действий или отправитель не имеет привилегий запуска фоновых действий. - Управляйте тем, как верхняя операция стека задач может завершить свою задачу . Если верхняя активность завершает задачу, Android вернется к той задаче, которая была активной в последний раз. Более того, если неосновное действие завершит свою задачу, Android вернется на главный экран; это не заблокирует завершение этого нетопового действия.
- Запретите запуск произвольных действий из других приложений в вашу собственную задачу . Это изменение не позволяет вредоносным приложениям фишинговать пользователей, создавая действия, выглядящие как действия других приложений.
- Блокировать невидимые окна для запуска фоновых действий . Это помогает предотвратить злоупотребление вредоносными приложениями запуска фоновой активности для отображения пользователям нежелательного или вредоносного контента.
Более безопасные намерения
Android 15 introduces new optional security measures to make intents safer and more robust. These changes are aimed at preventing potential vulnerabilities and misuse of intents that can be exploited by malicious apps. There are two main improvements to the security of intents in Android 15:
- Match target intent-filters: Intents that target specific components must accurately match the target's intent-filter specifications. If you send an intent to launch another app's activity, the target intent component needs to align with the receiving activity's declared intent-filters.
- Intents must have actions: Intents without an action will no longer match any intent-filters. This means that intents used to start activities or services must have a clearly defined action.
In order to check how your app responds to these changes, use
StrictMode
in your app. To see detailed
logs about Intent
usage violations, add the following method:
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
.

elegantTextHeight
для приложений, ориентированных на Android 14 (уровень API 34) и ниже. 
elegantTextHeight
для приложений, ориентированных на Android 15.Изменение ширины TextView для сложных форм букв
在以前的 Android 版本中,某些具有复杂形状的手写字体或语言可能会在上一个或下一个字符的区域绘制字母。在某些情况下,此类字母会在开头或结尾处被剪裁。从 Android 15 开始,TextView
会分配宽度,以便为此类字母绘制足够的空间,并允许应用请求向左额外添加内边距以防止剪裁。
由于此更改会影响 TextView
确定宽度的方式,因此如果应用以 Android 15(API 级别 35)或更高版本为目标平台,TextView
会默认分配更多宽度。您可以通过对 TextView
调用 setUseBoundsForWidth
API 来启用或停用此行为。
由于添加左内边距可能会导致现有布局未对齐,因此默认情况下不会添加内边距,即使以 Android 15 或更高版本为目标平台的应用也是如此。不过,您可以通过调用 setShiftDrawingOffsetForStartOverhang
添加额外的内边距以防止剪裁。
以下示例展示了这些更改如何改进某些字体和语言的文本布局。

<TextView android:fontFamily="cursive" android:text="java" />

<TextView android:fontFamily="cursive" android:text="java" android:useBoundsForWidth="true" android:shiftDrawingOffsetForStartOverhang="true" />

<TextView android:text="คอมพิวเตอร์" />

<TextView android:text="คอมพิวเตอร์" android:useBoundsForWidth="true" android:shiftDrawingOffsetForStartOverhang="true" />
Высота строки по умолчанию для EditText с учетом локали
В предыдущих версиях Android текстовый макет растягивал высоту текста так, чтобы она соответствовала высоте строки шрифта, соответствующего текущему языковому стандарту. Например, если контент был на японском языке, поскольку высота строки японского шрифта немного больше, чем у латинского шрифта, высота текста стала немного больше. Однако, несмотря на эти различия в высоте строк, размер элемента EditText
был одинаковым, независимо от используемой локали, как показано на следующем изображении:

EditText
, которые могут содержать текст на английском (en), японском (ja) и бирманском (my). Высота EditText
одинакова, хотя эти языки имеют разную высоту строк. Для приложений, ориентированных на Android 15 (уровень API 35), минимальная высота строки теперь зарезервирована для EditText
, чтобы соответствовать эталонному шрифту для указанного языкового стандарта, как показано на следующем изображении:

EditText
, которые могут содержать текст на английском (en), японском (ja) и бирманском (my). Высота EditText
теперь включает пространство для размещения высоты строки по умолчанию для шрифтов этих языков. При необходимости ваше приложение может восстановить предыдущее поведение, указав для атрибута useLocalePreferredLineHeightForMinimum
значение false
, а ваше приложение может установить собственные минимальные вертикальные метрики с помощью API setMinimumFontMetrics
в Kotlin и Java.
Камера и медиа
В 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 版本中的变更,请参阅 Android 15 中有关限制非 SDK 接口的更新。如需全面了解有关非 SDK 接口的详细信息,请参阅对非 SDK 接口的限制。