行為變更:指定 Android 15 以上版本的應用程式

和先前版本一樣,Android 15 也包含可能會影響應用程式的行為變更。以下行為變更僅適用於指定 Android 15 以上版本的應用程式。如果您的應用程式指定 Android 15 以上版本,建議您視情況修改應用程式,以支援這些行為。

此外,無論應用程式的 targetSdkVersion 為何,請務必查看影響所有在 Android 15 上執行的應用程式行為變更清單。

核心功能

Android 15 修改或擴充了 Android 系統的各種核心功能。

前景服務異動

我们将对 Android 15 中的前台服务进行以下更改。

数据同步前台服务超时行为

Android 15 為指定 Android 15 (API 級別 35) 以上版本為目標版本的應用程式,在 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]"

如要避免這項行為變更帶來的問題,您可以執行下列一或多項操作:

  1. 讓您的服務實作新的 Service.onTimeout(int, int) 方法。應用程式收到回呼時,請務必在幾秒內呼叫 stopSelf()。(如果您沒有立即停止應用程式,系統會產生失敗)。
  2. 確認應用程式的 dataSync 服務在任何 24 小時內的執行時間不超過 6 小時 (除非使用者與應用程式互動,並重設計時器)。
  3. 只有在使用者直接互動時,才啟動 dataSync 前景服務;由於服務啟動時應用程式處於前景,因此服務在應用程式進入背景後,可執行完整的六小時。
  4. 請改用其他 API,而非 dataSync 前景服務。

如果應用程式的 dataSync 前景服務在過去 24 小時內已執行 6 小時,則「除非」使用者將應用程式移至前景 (會重設計時器),否則您無法啟動另一項 dataSync 前景服務。如果您嘗試啟動另一項 dataSync 前景服務,系統會擲回 ForegroundServiceStartNotAllowedException,並顯示錯誤訊息「Time limit is Of for 前景服務類型 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 服務在 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]"

為避免例外狀況,您可以採取下列其中一種做法:

  1. 請讓服務實作新的 Service.onTimeout(int, int) 方法。應用程式收到回呼時,請務必在幾秒內呼叫 stopSelf()。(如果您沒有立即停止應用程式,系統會產生失敗)。
  2. 請確認應用程式的 mediaProcessing 服務在任何 24 小時內的總執行時間不超過 6 小時 (除非使用者與應用程式互動,重新設定計時器)。
  3. 只有在使用者直接互動時才啟動 mediaProcessing 前景服務;由於您的應用程式在服務啟動後位於前景,因此應用程式進入背景後,您的服務剩下六小時。
  4. 請改用 WorkManager 等其他 API,而非 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

对启动前台服务的 BOOT_COMPLETED 广播接收器的限制

在启动 BOOT_COMPLETED 广播接收器方面存在新限制 前台服务。BOOT_COMPLETED 接收器能启动 以下类型的前台服务:

如果 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 的调用(setInterruptionFiltersetNotificationPolicy)会导致创建或更新隐式 AutomaticZenRule,该 AutomaticZenRule 会根据这些 API 调用的调用周期开启和关闭。

请注意,只有当应用调用 setInterruptionFilter(INTERRUPTION_FILTER_ALL) 并希望该调用停用之前由其所有者激活的 AutomaticZenRule 时,此更改才会影响可观察到的行为。

OpenJDK API 變更

Android 15 持續更新 Android 核心程式庫,以便與最新版 OpenJDK LTS 中的功能保持一致。

如果應用程式指定 Android 15 (API 級別 35),以下部分變更可能會影響應用程式相容性:

  • 字串格式化 API 的變更:使用下列 String.format()Formatter.format() API 時,系統會更嚴格地驗證引數索引、旗標、寬度和精確度:

    舉例來說,如果使用引數索引 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() 相符。

在應用程式的建構設定中更新 compileSdk,改用 Android 15 (API 級別 35) 後,新的 SequencedCollection API 可能會影響應用程式的相容性:

  • kotlin-stdlib 中的 MutableList.removeFirst()MutableList.removeLast() 擴充函式發生衝突

    Java 中的 List 型別會對應至 Kotlin 中的 MutableList 型別。由於 Android 15 (API 級別 35) 中已導入 List.removeFirst()List.removeLast() API,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 中的其他方法發生衝突

    現有型別已新增方法,例如 ListDeque。這些新方法可能與其他介面和類別中,名稱和引數類型相同的方法不相容。如果方法簽章與不相容的項目發生衝突,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 不匹配的应用启动 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 使用違規的詳細記錄,請新增下列方法:

Kotlin


fun onCreate() {
    StrictMode.setVmPolicy(VmPolicy.Builder()
        .detectUnsafeIntentLaunch()
        .build()
    )
}

Java


public void onCreate() {
    StrictMode.setVmPolicy(new VmPolicy.Builder()
            .detectUnsafeIntentLaunch()
            .build());
}

使用者體驗和系統 UI

Android 15 包含一些變更,旨在打造更一致、直覺的使用者體驗。

視窗插邊變更

Android 15 有兩項與視窗內嵌相關的變更:預設會強制執行邊到邊,且有設定變更,例如系統列的預設設定。

全面实施政策

如果應用程式以 Android 15 (API 級別 35) 為目標版本,在搭載 Android 15 的裝置上預設會顯示無邊框模式。

以 Android 14 為目標版本,但未在 Android 15 裝置上採用無邊框模式的應用程式。


如果應用程式指定 Android 15 (API 級別 35) 為目標,且在 Android 15 裝置上採用無邊框設計,這個應用程式主要使用 Material 3 Compose 元件,可自動套用插邊。這個畫面不會受到 Android 15 無邊框強制措施的負面影響。

這項重大變更可能會對應用程式的 UI 造成負面影響。這項異動會影響下列 UI 區域:

  • 手勢控點導覽列
    • 預設為透明。
    • 底部位移已停用,因此除非套用插邊,否則內容會在系統導覽列後方繪製。
    • setNavigationBarColorR.attr#navigationBarColor 已淘汰,不會影響手勢操作模式。
    • setNavigationBarContrastEnforcedR.attr#navigationBarContrastEnforced 仍不會影響手勢操作模式。
  • 三按鈕操作
    • 預設不透明度為 80%,顏色可能與視窗背景相符。
    • 底部偏移已停用,因此內容會在系統導覽列後方繪製,除非套用插邊。
    • setNavigationBarColorR.attr#navigationBarColor 預設會設為與視窗背景相符。如要套用此預設值,視窗背景必須是顏色可繪項目。這個 API 已淘汰,但仍會影響三按鈕操作模式。
    • setNavigationBarContrastEnforcedR.attr#navigationBarContrastEnforced 預設為 true,這會在三按鈕操作列中加入 80% 不透明度的背景。
  • 狀態列
    • 預設為透明。
    • 頂端位移已停用,因此除非套用插邊,否則內容會在狀態列後方繪製。
    • setStatusBarColorR.attr#statusBarColor 已淘汰,對 Android 15 沒有影響。
    • setStatusBarContrastEnforcedR.attr#statusBarContrastEnforced 已淘汰,但仍會影響 Android 15。
  • 螢幕凹口
    • 非浮動視窗的 layoutInDisplayCutoutMode 必須為 LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYSSHORT_EDGESNEVERDEFAULT 會解讀為 ALWAYS,這樣使用者就不會看到因螢幕凹口而產生的黑邊,且畫面會顯示到螢幕邊框。

以下範例顯示應用程式在指定 Android 15 (API 級別 35) 前後,以及套用插邊前後的樣貌。

以 Android 14 為目標版本,但未在 Android 15 裝置上採用無邊框模式的應用程式。
如果應用程式指定 Android 15 (API 級別 35) 為目標,且在 Android 15 裝置上採用無邊框設計,不過,由於 Android 15 強制採用無邊框設計,許多元素現在會遭到狀態列、3 鍵導覽列或螢幕凹口遮蔽。隱藏的 UI 包括 Material 2 頂端應用程式列、懸浮動作按鈕和清單項目。
如果應用程式以 Android 15 (API 級別 35) 為目標,且在 Android 15 裝置上採用無邊框設計,並套用插邊,就不會隱藏 UI。
如果應用程式已採用無邊框設計,請檢查下列事項

如果您的應用程式已無邊框並套用插邊,則大多不會受到影響,但下列情況除外。不過,即使您認為應用程式不受影響,我們仍建議您進行測試。

  • 您有非浮動視窗,例如使用 SHORT_EDGESNEVERDEFAULT 而不是 LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYSActivity。如果應用程式在啟動時當機,可能是因為啟動畫面。您可以將核心啟動畫面依附元件升級至 1.2.0-alpha01 以上版本,或設定 window.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always
  • 可能會有流量較低的畫面,且使用者介面遭到遮蔽。確認這些較少造訪的畫面沒有遭到遮蔽的 UI。流量較低的螢幕包括:
    • 新手上路或登入畫面
    • 設定頁面
如果應用程式尚未採用無邊框設計,請檢查下列項目

如果您的應用程式尚未採用無邊框設計,很可能就會受到影響。除了已採用無邊框設計的應用程式情境,您也應考量下列事項:

  • 如果應用程式在 Compose 中使用 Material 3 元件 (androidx.compose.material3),例如 TopAppBarBottomAppBarNavigationBar,這些元件可能會不受影響,因為它們會自動處理插邊。
  • 如果應用程式使用 Compose 的 Material 2 元件 (androidx.compose.material),這類元件本身不會自動處理插邊。不過,您可以存取插邊,然後手動套用。在 androidx.compose.material 1.6.0 以上版本,請使用 windowInsets 參數,為 BottomAppBarTopAppBarBottomNavigationNavigationRail 手動套用插邊。同樣地,對 Scaffold 也是使用 contentWindowInsets 參數。
  • 如果應用程式使用 Views 和 Material 元件 (com.google.android.material),您可能不需採取額外行動,因為大多數以 Views 為基礎的 Material 元件 (例如 BottomNavigationViewBottomAppBarNavigationRailViewNavigationView) 都會處理插邊。不過,如果您使用 AppBarLayout,就需要新增 android:fitsSystemWindows="true"
  • 如果是自訂可組合函式,請手動將插邊套用為邊框間距。如果內容位於 Scaffold 內,您可以使用Scaffold 邊框間距值取用插邊。否則,請使用其中一個 WindowInsets 套用邊框間距。
  • 如果應用程式使用 Views 和 BottomSheetSideSheet 或自訂容器,請使用 ViewCompat.setOnApplyWindowInsetsListener 套用邊框間距。對於 RecyclerView,也請使用這個事件監聽器套用邊框間距,同時新增 clipToPadding="false"
如果應用程式必須提供自訂背景保護功能,請檢查下列事項

如果應用程式必須為三按鈕操作模式或狀態列提供自訂背景保護,應用程式應使用 WindowInsets.Type#tappableElement() 取得三按鈕操作模式導覽列高度或 WindowInsets.Type#statusBars,在系統列後方放置可組合函式或檢視區塊。

其他無邊框資源

如要進一步瞭解如何套用插邊,請參閱「無邊框檢視區塊」和「無邊框 Compose」指南。

已淘汰的 API

下列 API 已淘汰,但尚未停用:

下列 API 已停用:

稳定配置

如果應用程式指定 Android 15 (API 級別 35) 以上版本,Configuration系統不會再排除系統資訊列。如果您在 Configuration 類別中使用螢幕大小進行版面配置計算,請視需求改用適當的 ViewGroupWindowInsetsWindowMetricsCalculator 等更合適的替代方案。

Configuration 自 API 1 起就已推出,這項資訊通常來自 Activity.onConfigurationChanged。例如視窗密度、方向和大小。從 Configuration 傳回的視窗大小有一項重要特徵,就是先前會排除系統資訊列。

設定大小通常用於資源選取 (例如 /res/layout-h500dp),這仍是有效的用途。不過,我們一直不建議使用這項屬性計算版面配置。如果這樣做,請立即遠離該裝置。視用途而定,您應將 Configuration 替換為更合適的項目。

如果使用它來計算版面配置,請使用適當的 ViewGroup,例如 CoordinatorLayoutConstraintLayout。如要使用這項屬性判斷系統導覽列的高度,請使用 WindowInsets。如要瞭解應用程式視窗的目前大小,請使用 computeCurrentWindowMetrics

下列清單說明這項異動會影響的欄位:

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 新增額外的邊框間距,避免發生裁剪情形。

以下範例說明這些變更如何改善部分字型和語言的文字版面配置。

以草書字型顯示的英文文字標準版面配置。部分字母會遭到裁剪。以下是對應的 XML:

<TextView
    android:fontFamily="cursive"
    android:text="java" />
相同英文文字的版面配置,具有額外寬度和邊框間距。以下是對應的 XML:

<TextView
    android:fontFamily="cursive"
    android:text="java"
    android:useBoundsForWidth="true"
    android:shiftDrawingOffsetForStartOverhang="true" />
泰文的標準版面配置。部分字母遭到裁剪。 以下是對應的 XML:

<TextView
    android:text="คอมพิวเตอร์" />
相同泰文文字的版面配置,並增加寬度和邊框間距。以下是對應的 XML:

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

EditText 的預設行高會因應語言代碼而異

在较低版本的 Android 中,文本布局会拉伸文本的高度,以满足与当前语言区域匹配的字体的行高。例如,如果内容是日语,由于日语字体的行高略高于拉丁字体,因此文本的高度会略高。不过,尽管行高存在这些差异,但无论使用的是哪种语言区域,EditText 元素的大小都是统一的,如下图所示:

三个框,表示可以包含英语 (en)、日语 (ja) 和缅甸语 (my) 文本的 EditText 元素。EditText 的高度相同,即使这些语言的行高各不相同。

对于以 Android 15(API 级别 35)为目标平台的应用,现在为 EditText 预留了最小行高,以匹配指定语言区域的参考字体,如下图所示:

三个框,表示可以包含英语 (en)、日语 (ja) 和缅甸语 (my) 文本的 EditText 元素。EditText 的高度现在包含足够的空间来容纳这些语言字体的默认行高。

如有需要,您的应用可以将 useLocalePreferredLineHeightForMinimum 属性指定为 false,以恢复之前的行为;您的应用还可以在 Kotlin 和 Java 中使用 setMinimumFontMetrics API 设置自定义最小垂直指标。

相機和媒體

Android 15 對指定 Android 15 以上版本的應用程式,進行了下列攝影機和媒體行為變更。

要求音訊焦點的限制

以 Android 15(API 级别 35)为目标平台的应用必须是顶部应用或正在运行前台服务,才能请求音频焦点。如果应用在未满足上述任一要求的情况下尝试请求焦点,调用将返回 AUDIOFOCUS_REQUEST_FAILED

如需详细了解音频焦点,请参阅管理音频焦点

更新非 SDK 限制

基於與 Android 開發人員合作及最新的內部測試,Android 15 包含更新後的受限制非 SDK 介面清單。在限制非 SDK 介面之前,我們盡可能確保公開替代方案的可得性。

如果您的應用程式並不是以 Android 15 為目標版本,則此處所述的某些變更可能不會立即對您造成影響。不過,應用程式視目標 API 級別而定,可能可以存取某些非 SDK 介面,但使用任何非 SDK 方法或欄位時,應用程式停止運作的風險極高。

如果不確定應用程式是否使用非 SDK 介面,可測試應用程式以便確認。如果應用程式仰賴非 SDK 介面,則應開始規劃遷移至 SDK 替代方案。不過我們瞭解,有些應用程式可使用非 SDK 介面運作。如果您除了為應用程式中的某個功能使用非 SDK 介面外,已別無他法,則應要求新的公用 API

如需详细了解此 Android 版本中的变更,请参阅 Android 15 中有关限制非 SDK 接口的更新。如需全面了解有关非 SDK 接口的详细信息,请参阅对非 SDK 接口的限制