Como nas versões anteriores, o Android 15 inclui mudanças de comportamento que podem afetar seu app. As seguintes mudanças de comportamento se aplicam exclusivamente a apps destinados ao Android 15 ou mais recente. Caso seu app seja direcionado ao Android 15 ou a versões mais recentes, faça modificações para oferecer suporte a esses comportamentos de forma adequada, quando aplicável.
Consulte também a lista de mudanças de comportamento que afetam todos os apps
executados no Android 15, independente da targetSdkVersion
do app.
Principal recurso
O Android 15 modifica ou expande vários recursos principais do sistema Android.
Mudanças nos serviços em primeiro plano
Estamos fazendo as seguintes mudanças nos serviços em primeiro plano com o Android 15.
- Comportamento de tempo limite do serviço em primeiro plano de sincronização de dados
- Novo tipo de serviço em primeiro plano de processamento de mídia
- Restrições para broadcast receivers
BOOT_COMPLETED
que iniciam serviços em primeiro plano - Restrições para iniciar serviços em primeiro plano enquanto um app tem a permissão
SYSTEM_ALERT_WINDOW
Comportamento de tempo limite do serviço em primeiro plano da sincronização de dados
对于以 Android 15(API 级别 35)或更高版本为目标平台的应用,Android 15 为 dataSync
引入了新的超时行为。此行为也适用于新的 mediaProcessing
前台服务类型。
系统允许应用的 dataSync
服务在 24 小时内共运行 6 小时,之后系统会调用正在运行的服务的 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
服务在任何 24 小时内总运行时间不超过 6 小时(除非用户与应用互动,重置计时器)。 - 仅在有直接用户互动时启动
dataSync
前台服务;由于服务启动时应用位于前台,因此您的服务在应用进入后台后有完整的 6 小时时间。 - 请使用替代 API,而不是使用
dataSync
前台服务。
如果您的应用的 dataSync
前台服务在过去 24 小时内运行了 6 小时,则您无法启动其他 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
Novo tipo de serviço em primeiro plano de processamento de mídia
Android 15 引入了一种新的前台服务类型 mediaProcessing
。此服务类型适用于转码媒体文件等操作。例如,媒体应用可能会下载音频文件,并需要先将其转换为其他格式,然后才能播放。您可以使用 mediaProcessing
前台服务,确保即使应用在后台运行时转换也会继续。
系统允许应用的 mediaProcessing
服务在 24 小时内总共运行 6 小时,之后系统会调用正在运行的服务的 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
服务在任何 24 小时内总运行时间不超过 6 小时(除非用户与应用互动,重置计时器)。 - 仅在有直接用户互动时启动
mediaProcessing
前台服务;由于服务启动时应用位于前台,因此您的服务在应用进入后台后有完整的 6 小时时间。 - 请改用 替代 API(例如 WorkManager),而不是使用
mediaProcessing
前台服务。
如果您的应用的 mediaProcessing
前台服务在过去 24 小时内运行了 6 小时,则您无法启动其他 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
Restrições em broadcast receivers BOOT_COMPLETED
que iniciam serviços em primeiro plano
在启动 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
Restrições para iniciar serviços em primeiro plano enquanto um app tem a permissão 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
Mudanças na hora em que os apps podem modificar o estado global do modo "Não perturbe"
Os apps direcionados ao Android 15 (nível 35 da API) e mais recentes não podem mais mudar o
estado global ou a política de Não perturbe (DND, na sigla em inglês) em um dispositivo, seja modificando
as configurações do usuário ou desativando o modo DND. Em vez disso, os apps precisam contribuir com um
AutomaticZenRule
, que o sistema combina em uma política global com o
esquema de política mais restritiva. As chamadas para APIs que
afetam o estado global (setInterruptionFilter
,
setNotificationPolicy
) resultam na criação ou atualização de um
AutomaticZenRule
implícito, que é ativado e desativado dependendo do ciclo de chamadas
dessas chamadas de API.
Essa mudança só afeta o comportamento observável se o app estiver chamando
setInterruptionFilter(INTERRUPTION_FILTER_ALL)
e esperar que essa chamada desative uma AutomaticZenRule
que foi ativada anteriormente pelos proprietários.
Mudanças na API OpenJDK
Android 15 将继续更新 Android 的核心库,以与最新 OpenJDK LTS 版本中的功能保持一致。
其中一些变更可能会影响以 Android 15(API 级别 35)为目标平台的应用的兼容性:
对字符串格式化 API 的更改:现在,使用以下
String.format()
和Formatter.format()
API 时,对参数索引、标志、宽度和精度的验证更为严格: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
)。对随机 int 序列的更改:在 https://bugs.openjdk.org/browse/JDK-8301574 中进行更改后,以下
Random.ints()
方法现在返回的数字序列与Random.nextInt()
方法返回的序列不同:通常,此更改不会导致应用出现破坏行为,但您的代码不应预期从
Random.ints()
方法生成的序列与Random.nextInt()
匹配。
在您更新应用 build 配置中的 compileSdk
以使用 Android 15(API 级别 35)后,新的 SequencedCollection
API 可能会影响应用的兼容性:
与
kotlin-stdlib
中的MutableList.removeFirst()
和MutableList.removeLast()
扩展函数发生冲突Java 中的
List
类型会映射到 Kotlin 中的MutableList
类型。由于List.removeFirst()
和List.removeLast()
API 已在 Android 15(API 级别 35)中引入,因此 Kotlin 编译器会将函数调用(例如list.removeFirst()
)静态解析为新的List
API,而不是kotlin-stdlib
中的扩展函数。如果将应用重新编译并将
compileSdk
设置为35
且将minSdk
设置为34
或更低级别,然后在 Android 14 及更低版本上运行该应用,系统会抛出运行时错误:java.lang.NoSuchMethodError: No virtual method removeFirst()Ljava/lang/Object; in class Ljava/util/ArrayList;
Android Gradle 插件中现有的
NewApi
lint 选项可以捕获这些新的 API 用法。./gradlew lint
MainActivity.kt:41: Error: Call requires API level 35 (current min is 34): java.util.List#removeFirst [NewApi] list.removeFirst()如需修复运行时异常和 lint 错误,可以在 Kotlin 中将
removeFirst()
和removeLast()
函数调用分别替换为removeAt(0)
和removeAt(list.lastIndex)
。如果您使用的是 Android Studio Ladybug | 2024.1.3 或更高版本,则该版本还提供了针对这些错误的快速修复选项。如果 lint 选项已停用,请考虑移除
@SuppressLint("NewApi")
和lintOptions { disable 'NewApi' }
。与 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(); }
Segurança
O Android 15 inclui mudanças que promovem a segurança do sistema para ajudar a proteger apps e usuários contra apps maliciosos.
Versões TLS restritas
Android 15 限制了对 TLS 版本 1.0 和 1.1 的使用。这些版本之前已在 Android 中被弃用,但现在不允许面向 Android 15 的应用使用。
Inicia atividades seguras em segundo plano
O Android 15 protege os usuários contra apps maliciosos e dá a eles mais controle sobre dos dispositivos deles adicionando alterações que evitam que apps maliciosos em segundo plano colocar outros apps em primeiro plano, aumentar os privilégios deles e abusar interação do usuário. Os lançamentos de atividades em segundo plano estão restritos desde Android 10 (nível 29 da API)
Impedir que apps que não correspondam ao UID superior da pilha iniciem atividades
Apps maliciosos podem iniciar a atividade de outro app na mesma tarefa e depois
se sobrepõem, criando a ilusão de ser o app. Essa tarefa
sequestrando" contorna as restrições atuais de inicialização em segundo plano porque tudo
ocorre dentro da mesma tarefa visível. Para mitigar esse risco, o Android 15 adiciona uma
flag que impede a inicialização de apps que não correspondem ao UID superior da pilha
atividades. Para ativar todas as atividades do seu app, atualize o
allowCrossUidActivitySwitchFromBelow
no arquivo AndroidManifest.xml
do app:
<application android:allowCrossUidActivitySwitchFromBelow="false" >
As novas medidas de segurança serão ativadas se todas as condições a seguir forem verdadeiras:
- O app que executa a inicialização é destinado ao Android 15.
- O app na parte de cima da pilha de tarefas é destinado ao Android 15.
- Qualquer atividade visível ativou as novas proteções
Se as medidas de segurança estiverem ativadas, os apps poderão voltar à tela inicial em vez último app visível, se terminarem a própria tarefa.
Outras mudanças
Além da restrição para correspondência de UID, essas outras mudanças também são incluído:
- Mude
PendingIntent
criadores de conteúdo para bloquear o lançamento de atividades em segundo plano ao padrão. Isso ajuda a evitar que os apps criem acidentalmente umPendingIntent
que podem ser usados indevidamente por usuários maliciosos. - Não levar um app para o primeiro plano, a menos que o remetente
PendingIntent
permite. O objetivo dessa mudança é impedir que apps maliciosos iniciar atividades em segundo plano. Por padrão, os apps não são tem permissão para colocar a pilha de tarefas em primeiro plano, a menos que o criador permita privilégios de inicialização de atividade em segundo plano ou o remetente tem atividade em segundo plano iniciar privilégios. - Controlar como a principal atividade de uma pilha de tarefas pode concluir a tarefa. Se o principal atividade finaliza uma tarefa, o Android volta para a tarefa que ativo pela última vez. Além disso, se uma atividade não principal terminar a tarefa, o Android voltar para a tela inicial não vai bloquear o final desse objeto atividades.
- Impedir a inicialização de atividades arbitrárias de outros apps no seu próprio app tarefa. Essa mudança impede que apps maliciosos façam phishing pelos usuários criando atividades que parecem ser de outros apps.
- Impeça que janelas não visíveis sejam consideradas para atividades em segundo plano é lançado. Isso ajuda a evitar que apps maliciosos usem o plano de fundo de atividades para exibir conteúdo indesejado ou malicioso aos usuários.
Intents mais seguras
Android 15 引入了新的可选安全措施,以提高 intent 的安全性和稳健性。这些变更旨在防止潜在的漏洞以及恶意应用可能利用的 intent 滥用行为。Android 15 对 intent 的安全性进行了两项主要改进:
- 与目标 intent 过滤器匹配:定位到特定组件的 intent 必须与目标的 intent 过滤器规范完全匹配。如果您发送 intent 来启动其他应用的 activity,目标 intent 组件需要与接收 activity 声明的 intent 过滤器保持一致。
- intent 必须具有操作:没有操作的 intent 将不再与任何 intent 过滤器匹配。这意味着,用于启动 activity 或服务的 intent 必须具有明确定义的操作。
如需检查您的应用对这些更改的响应方式,请在应用中使用 StrictMode
。如需查看有关 Intent
使用违规行为的详细日志,请添加以下方法:
Kotlin
fun onCreate() { StrictMode.setVmPolicy(VmPolicy.Builder() .detectUnsafeIntentLaunch() .build() ) }
Java
public void onCreate() { StrictMode.setVmPolicy(new VmPolicy.Builder() .detectUnsafeIntentLaunch() .build()); }
Experiência do usuário e interface do sistema
O Android 15 inclui algumas mudanças que têm como objetivo criar uma experiência do usuário mais consistente e intuitiva.
Mudanças no encarte da janela
Android 15 中与窗口内边距相关的两项变更:默认强制执行边到边,此外还有配置变更,例如系统栏的默认配置。
全面实施政策
Os apps são exibidos de ponta a ponta por padrão em dispositivos com o Android 15 se o app for destinado ao Android 15 (nível 35 da API).

Essa é uma mudança importante que pode afetar negativamente a interface do seu app. As mudanças afetam as seguintes áreas da interface:
- Barra de navegação do identificador de gestos
- Transparente por padrão.
- O deslocamento inferior está desativado para que o conteúdo seja mostrado por trás da barra de navegação do sistema, a menos que os insets sejam aplicados.
setNavigationBarColor
eR.attr#navigationBarColor
foram descontinuados e não afetam a navegação por gestos.setNavigationBarContrastEnforced
eR.attr#navigationBarContrastEnforced
continuam sem efeito na navegação por gestos.
- Navegação com três botões
- Opacidade definida como 80% por padrão, com a cor possivelmente correspondendo ao plano de fundo da janela.
- O deslocamento da parte de baixo foi desativado para que o conteúdo apareça por trás da barra de navegação do sistema, a menos que os insets sejam aplicados.
setNavigationBarColor
eR.attr#navigationBarColor
são definidos para corresponder ao plano de fundo da janela por padrão. O plano de fundo da janela precisa ser um drawable de cor para que esse padrão seja aplicado. Essa API foi descontinuada, mas continua afetando a navegação com três botões.setNavigationBarContrastEnforced
eR.attr#navigationBarContrastEnforced
são verdadeiros por padrão, o que adiciona um plano de fundo opaco de 80% à navegação com três botões.
- Barra de status
- Transparente por padrão.
- O deslocamento superior está desativado para que o conteúdo seja renderizado atrás da barra de status, a menos que os encartes sejam aplicados.
setStatusBarColor
eR.attr#statusBarColor
foram descontinuados e não têm efeito no Android 15.setStatusBarContrastEnforced
eR.attr#statusBarContrastEnforced
foram descontinuados, mas ainda têm efeito no Android 15.
- Recorte da tela
- O
layoutInDisplayCutoutMode
de janelas não flutuantes precisa serLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
.SHORT_EDGES
,NEVER
eDEFAULT
são interpretados comoALWAYS
para que os usuários não vejam uma barra preta causada pelo recorte da tela e apareçam de borda a borda.
- O
O exemplo a seguir mostra um app antes e depois de segmentar o Android 15 (nível 35 da API) e antes e depois de aplicar insets.



Como verificar se o app já ocupa toda a tela
Se o app já ocupa toda a tela e aplica insets, não há grande impacto, exceto nos seguintes cenários. No entanto, mesmo que você ache que não foi afetado, recomendamos testar o app.
- Você tem uma janela não flutuante, como uma
Activity
que usaSHORT_EDGES
,NEVER
ouDEFAULT
em vez deLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
. Se o app falhar na inicialização, isso pode ser devido à tela de apresentação. É possível fazer upgrade da dependência da tela de inicialização do núcleo para 1.2.0-alpha01 ou versões mais recentes ou definirwindow.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always
. - Talvez haja telas com menos tráfego e uma interface obstruída. Verifique se essas
telas menos visitadas não têm uma interface oculta. As telas com menos tráfego incluem:
- Telas de integração ou de login
- Páginas de configurações
O que verificar se o app ainda não ocupa toda a tela
Se o app ainda não ocupa toda a tela, é provável que você seja afetado. Além dos cenários para apps que já ocupam toda a tela, considere o seguinte:
- Se o app usar componentes do Material 3 (
androidx.compose.material3
) no Compose, comoTopAppBar
,BottomAppBar
eNavigationBar
, esses componentes provavelmente não serão afetados porque processam encartes automaticamente. - Se o app estiver usando componentes do Material 2 (
androidx.compose.material
) no Compose, eles não vão processar encartes automaticamente. No entanto, você pode conseguir acesso aos encartes e aplicá-los manualmente. Na androidx.compose.material 1.6.0 e versões mais recentes, use o parâmetrowindowInsets
para aplicar os encartes manualmente paraBottomAppBar
,TopAppBar
,BottomNavigation
eNavigationRail
. Da mesma forma, use o parâmetrocontentWindowInsets
paraScaffold
. - Caso o app use visualizações e componentes do Material
(
com.google.android.material
), a maioria dos componentes do Material baseados em visualizações, comoBottomNavigationView
,BottomAppBar
,NavigationRailView
ouNavigationView
, processa encartes e não requer trabalho extra. No entanto, é necessário adicionarandroid:fitsSystemWindows="true"
se estiver usandoAppBarLayout
. - Para elementos combináveis personalizados, aplique os encartes manualmente como padding. Se o
conteúdo estiver em um
Scaffold
, você poderá consumir insets usando os valores de paddingScaffold
. Caso contrário, aplique o padding usando um dosWindowInsets
. - Se o app estiver usando visualizações e
BottomSheet
,SideSheet
ou contêineres personalizados, aplique padding usandoViewCompat.setOnApplyWindowInsetsListener
. ParaRecyclerView
, aplique padding usando esse listener e também adicioneclipToPadding="false"
.
O que verificar se o app precisa oferecer proteção personalizada em segundo plano
Se o app precisar oferecer proteção em segundo plano personalizada para a navegação de três botões ou
a barra de status, ele precisará colocar um elemento combinável ou uma visualização atrás da barra do sistema
usando WindowInsets.Type#tappableElement()
para conseguir a altura da barra de navegação
de três botões ou WindowInsets.Type#statusBars
.
Outros recursos de ponta a ponta
Consulte os guias Visualizações de borda a borda e Composição de borda a borda para mais considerações sobre como aplicar insets.
APIs descontinuadas
As APIs a seguir foram descontinuadas, mas não foram desativadas:
R.attr#enforceStatusBarContrast
R.attr#navigationBarColor
(para navegação com três botões, com 80% de alfa)Window#isStatusBarContrastEnforced
Window#setNavigationBarColor
(para navegação com três botões, com 80% de alfa)Window#setStatusBarContrastEnforced
As APIs a seguir foram descontinuadas e desativadas:
R.attr#navigationBarColor
(para navegação por gestos)R.attr#navigationBarDividerColor
R.attr#statusBarColor
Window#setDecorFitsSystemWindows
Window#getNavigationBarColor
Window#getNavigationBarDividerColor
Window#getStatusBarColor
Window#setNavigationBarColor
(para navegação por gestos)Window#setNavigationBarDividerColor
Window#setStatusBarColor
稳定配置
Se o app for direcionado ao Android 15 (nível 35 da API) ou mais recente, o Configuration
não
mais vai excluir as barras do sistema. Se você usar o tamanho da tela na
classe Configuration
para o cálculo do layout, substitua-o por melhores
alternativas, como ViewGroup
, WindowInsets
ou
WindowMetricsCalculator
, dependendo da sua necessidade.
Configuration
está disponível desde a API 1. Ele geralmente é recebido de
Activity.onConfigurationChanged
. Ele fornece informações como densidade de janela,
orientação e tamanhos. Uma característica importante sobre os tamanhos de janela
retornados de Configuration
é que eles excluíam as barras do sistema.
O tamanho da configuração geralmente é usado para seleção de recursos, como
/res/layout-h500dp
, e ainda é um caso de uso válido. No entanto, o uso dele para
cálculo de layout sempre foi desencorajado. Se for o caso, afaste-se
dele agora. Substitua o uso de Configuration
por algo
mais adequado, dependendo do seu caso de uso.
Se você usar o método para calcular o layout, use um ViewGroup
adequado, como
CoordinatorLayout
ou ConstraintLayout
. Se você usar para determinar a altura
da barra de navegação do sistema, use WindowInsets
. Se você quiser saber o tamanho atual
da janela do app, use computeCurrentWindowMetrics
.
A lista a seguir descreve os campos afetados por essa mudança:
- Os tamanhos
Configuration.screenWidthDp
escreenHeightDp
não mais excluem as barras do sistema. Configuration.smallestScreenWidthDp
é afetado indiretamente por mudanças emscreenWidthDp
escreenHeightDp
.Configuration.orientation
é afetado indiretamente por mudanças emscreenWidthDp
escreenHeightDp
em dispositivos próximos ao quadrado.Display.getSize(Point)
é afetado indiretamente pelas mudanças emConfiguration
. Esse recurso foi descontinuado a partir do nível 30 da API.- O
Display.getMetrics()
já funcionava assim desde o nível 33 da API.
O atributo "elegantTextHeight" tem o padrão definido como "true".
Em apps destinados ao Android 15 (nível 35 da API), o atributo
elegantTextHeight
TextView
passa a ser true
por padrão, substituindo a fonte compacta usada por padrão por alguns
scripts com métricas verticais grandes por uma que é muito mais legível.
A fonte compacta foi introduzida para evitar a quebra de layouts. O Android 13 (nível
33 da API) evita muitas dessas quebras, permitindo que o layout de texto
estique a altura vertical usando o atributo
fallbackLineSpacing
.
No Android 15, a fonte compacta ainda permanece no sistema. Assim, o app pode definir
elegantTextHeight
como false
para ter o mesmo comportamento de antes, mas é
improvável que haja suporte a ele nas próximas versões. Portanto, se o app oferecer suporte aos
seguintes scripts: árabe, lao, birmanês, tâmil, gujarati, canarês, malaiala,
odia, télugo ou tailandês, teste o app definindo elegantTextHeight
como true
.

elegantTextHeight
comportamento para apps destinados ao Android 14 (nível 34 da API) e versões anteriores.
elegantTextHeight
para apps destinados ao Android 15.A largura do TextView muda para formas de letras complexas
在以前的 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" />
Altura de linha padrão compatível com a localidade para EditText
Nas versões anteriores do Android, o layout de texto estendia a altura do
texto para atender à altura da linha da fonte que correspondia à localidade atual. Por
exemplo, se o conteúdo estivesse em japonês, a altura da linha da fonte japonesa
seria um pouco maior do que a de uma fonte latina, e a altura do texto
ficaria um pouco maior. No entanto, apesar dessas diferenças nas alturas das linhas, o elemento
EditText
foi dimensionado de maneira uniforme, independentemente
da localidade usada, conforme ilustrado na imagem a seguir:

EditText
que
podem conter texto em inglês (en), japonês (ja) e birmanês (my). A
altura do EditText
é a mesma, mesmo que esses idiomas
tenham alturas de linha diferentes.Para apps direcionados ao Android 15 (nível 35 da API), uma altura mínima de linha agora é
reservada para EditText
para corresponder à fonte de referência da localidade especificada, conforme
mostrado na imagem a seguir:

EditText
que
podem conter texto em inglês (en), japonês (ja) e birmanês (my). A
altura do EditText
agora inclui espaço para acomodar a
altura da linha padrão das fontes desses idiomas.Se necessário, o app pode restaurar o comportamento anterior especificando o atributo
useLocalePreferredLineHeightForMinimum
para false
e definir métricas verticais mínimas personalizadas usando a
API setMinimumFontMetrics
em Kotlin e Java.
Câmera e mídia
O Android 15 faz as seguintes mudanças no comportamento da câmera e da mídia para apps destinados ao Android 15 ou mais recente.
Restrições ao solicitar o foco de áudio
以 Android 15(API 级别 35)为目标平台的应用必须是顶部应用或正在运行前台服务,才能请求音频焦点。如果应用在未满足上述任一要求的情况下尝试请求焦点,调用将返回 AUDIOFOCUS_REQUEST_FAILED
。
如需详细了解音频焦点,请参阅管理音频焦点。
Atualização das restrições não SDK
O Android 15 inclui listas atualizadas de interfaces não SDK restritas com base na colaboração de desenvolvedores do Android e nos testes internos mais recentes. Antes de restringirmos interfaces não SDK, sempre que possível, garantimos que haja alternativas públicas disponíveis.
Caso seu app não seja destinado ao Android 15, é possível que algumas dessas mudanças não afetem você imediatamente. No entanto, embora seja possível que o app acesse algumas interfaces não SDK dependendo do nível da API de destino do app, o uso de qualquer método ou campo não SDK sempre apresenta um alto risco de corromper o app.
Se você não sabe se o app usa interfaces não SDK, teste-o para descobrir. Se o app depende de interfaces não SDK, comece a planejar uma migração para alternativas SDK. No entanto, entendemos que alguns apps têm casos de uso válidos para interfaces que não são do SDK. Se você não encontrar uma alternativa para deixar de usar uma interface fora do SDK em um recurso no app, solicite uma nova API pública.
如需详细了解此 Android 版本中的变更,请参阅 Android 15 中有关限制非 SDK 接口的更新。如需全面了解有关非 SDK 接口的详细信息,请参阅对非 SDK 接口的限制。