行為變更:指定 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

應用程式可修改「零打擾」模式全域狀態的時間異動

Apps that target Android 15 (API level 35) and higher can no longer change the global state or policy of Do Not Disturb (DND) on a device (either by modifying user settings, or turning off DND mode). Instead, apps must contribute an AutomaticZenRule, which the system combines into a global policy with the existing most-restrictive-policy-wins scheme. Calls to existing APIs that previously affected global state (setInterruptionFilter, setNotificationPolicy) result in the creation or update of an implicit AutomaticZenRule, which is toggled on and off depending on the call-cycle of those API calls.

Note that this change only affects observable behavior if the app is calling setInterruptionFilter(INTERRUPTION_FILTER_ALL) and expects that call to deactivate an AutomaticZenRule that was previously activated by their owners.

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 匹配限制之外,还包括以下其他更改:

  • 默认情况下,将 PendingIntent 创建者更改为屏蔽后台活动启动。这有助于防止应用意外创建可能会被恶意操作者滥用的 PendingIntent
  • 除非 PendingIntent 发件人允许,否则请勿将应用调至前台。此变更旨在防止恶意应用滥用 在后台启动 activity 的功能。默认情况下,应用 允许将任务堆栈转到前台,除非创建者允许 后台活动启动权限或发送者有后台活动 启动权限
  • 控制任务堆栈顶部 activity 如何完成其任务。如果顶部 activity 完成任务,Android 将返回上次处于活动状态的任务。此外,如果非顶层 activity 完成其任务,Android 将 返回主屏幕;它不会挡住这个非顶层的 活动。
  • 防止将其他应用中的任意 activity 启动到您自己的 activity 任务。这项变更可防止恶意应用通过创建看似来自其他应用的 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 Auto 上的顯示方式可能有所不同。

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

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

  • 您有非浮動視窗,例如使用 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 以测试您的应用。

针对以 Android 14(API 级别 34)及更低版本为目标平台的应用的 elegantTextHeight 行为。
以 Android 15 为目标平台的应用的 elegantTextHeight 行为。

複雜字母形狀的 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 元素的大小仍保持一致,不受使用語言代碼影響,如下圖所示:

三個方塊代表 EditText 元素,可包含英文 (en)、日文 (ja) 和緬甸文 (my) 的文字。EditText 的高度相同,即使這些語言的行高不同。

針對指定 Android 15 (API 級別 35) 的應用程式,現在會為 EditText 保留最小行高,以便與指定語言代碼的參考字型相符,如以下圖片所示:

三個方塊代表 EditText 元素,可包含英文 (en)、日文 (ja) 和緬甸文 (my) 的文字。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 版本中的變更,請參閱「Android 15 的非 SDK 介面限制更新內容」。如要進一步瞭解非 SDK 介面的一般資訊,請參閱「非 SDK 介面的限制」。