Thay đổi về hành vi: Ứng dụng nhắm đến Android 15 trở lên

Giống như các bản phát hành trước, Android 15 có các thay đổi về hành vi có thể ảnh hưởng đến ứng dụng của bạn. Những thay đổi sau đây về hành vi chỉ áp dụng cho các ứng dụng nhắm đến Android 15 trở lên. Nếu ứng dụng của bạn nhắm đến Android 15 trở lên, bạn nên sửa đổi ứng dụng của mình để hỗ trợ những hành vi này một cách phù hợp, khi có thể áp dụng.

Ngoài ra, hãy nhớ tham khảo danh sách thay đổi về hành vi ảnh hưởng đến tất cả ứng dụng chạy trên Android 15 bất kể targetSdkVersion của ứng dụng là gì.

Chức năng cốt lõi

Android 15 sửa đổi hoặc mở rộng nhiều chức năng cốt lõi của hệ thống Android.

Các thay đổi đối với dịch vụ trên nền trước

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

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

Android 15 ra mắt hành vi hết thời gian chờ mới cho dataSync đối với việc nhắm mục tiêu ứng dụng Android 15 (API cấp 35) trở lên. Hành vi này cũng áp dụng cho Loại dịch vụ trên nền trước mediaProcessing.

Hệ thống này cho phép các dịch vụ dataSync của một ứng dụng chạy tổng cộng 6 giờ trong khoảng thời gian 24 giờ, sau đó hệ thống sẽ gọi lệnh gọi Phương thức Service.onTimeout(int, int) (được giới thiệu trong Android 15). Vào thời điểm này, dịch vụ có vài giây để gọi Service.stopSelf(). Khi Service.onTimeout() được gọi, không còn được coi là dịch vụ trên nền trước. Nếu dịch vụ không gọi Service.stopSelf(), hệ thống sẽ gửi ra một ngoại lệ nội bộ. Chiến lược phát hành đĩa đơn trường hợp ngoại lệ được ghi lại trong Logcat với thông báo sau:

Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type dataSync did not stop within its timeout: [component name]"

Để tránh các vấn đề với thay đổi này về hành vi, bạn có thể thực hiện một hoặc nhiều cách sau:

  1. Yêu cầu dịch vụ của bạn triển khai phương thức Service.onTimeout(int, int) mới. Khi ứng dụng của bạn nhận được lệnh gọi lại, hãy nhớ gọi stopSelf() trong một vài giây. (Nếu bạn không dừng ứng dụng ngay lập tức, hệ thống sẽ tạo một lỗi.)
  2. Đảm bảo các dịch vụ dataSync của ứng dụng không chạy tổng cộng 6 giờ trong khoảng thời gian 24 giờ bất kỳ (trừ phi người dùng tương tác với ứng dụng), đang đặt lại đồng hồ hẹn giờ).
  3. Chỉ bắt đầu dataSync dịch vụ trên nền trước do người dùng trực tiếp tương tác; vì ứng dụng của bạn chạy ở nền trước khi dịch vụ bắt đầu, thì dịch vụ của bạn có đủ 6 giờ kể từ khi ứng dụng chuyển sang chạy ở chế độ nền.
  4. Thay vì dùng dịch vụ dataSync trên nền trước, hãy dùng API thay thế.

Nếu dịch vụ trên nền trước dataSync của ứng dụng đã chạy được 6 giờ qua 24, bạn không thể bắt đầu một dịch vụ trên nền trước dataSync khác trừ phi người dùng đã đưa ứng dụng của bạn lên nền trước (đặt lại bộ tính giờ). Nếu bạn cố gắng bắt đầu một dịch vụ trên nền trước dataSync khác, thì hệ thống sẽ gửi ForegroundServiceStartNotAllowedException kèm theo thông báo lỗi như "Đã hết giới hạn thời gian cho dịch vụ trên nền trước nhập dataSync".

Thử nghiệm

Để kiểm thử hành vi của ứng dụng, bạn có thể bật tính năng thời gian chờ đồng bộ hoá dữ liệu ngay cả khi ứng dụng không nhắm đến Android 15 (miễn là ứng dụng đó đang chạy trên Android 15 thiết bị). Để bật tính năng thời gian chờ, hãy chạy lệnh adb sau:

adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name

Bạn cũng có thể điều chỉnh khoảng thời gian chờ để dễ dàng thử nghiệm cách ứng dụng sẽ hoạt động khi đạt đến giới hạn. Để đặt khoảng thời gian chờ mới, hãy chạy sau đây là lệnh 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:

  1. Have your service implement the new Service.onTimeout(int, int) method. When your app receives the callback, make sure to call stopSelf() within a few seconds. (If you don't stop the app right away, the system generates a failure.)
  2. 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).
  3. 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.
  4. 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 接收器能启动 以下类型的前台服务:

如果 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

Thay đổi về thời điểm các ứng dụng có thể sửa đổi trạng thái chung của chế độ Không làm phiền

以 Android 15 为目标平台的应用无法再更改设备上的全局状态或勿扰 (DND) 政策(通过修改用户设置或关闭 DND 模式)。相反,应用必须提供一个 AutomaticZenRule,系统会将后者合并到一个具有现有最严格的政策胜出方案的全局政策中。调用之前影响全局状态的现有 API(setInterruptionFiltersetNotificationPolicy)会导致创建或更新隐式 AutomaticZenRule,该 AutomaticZenRule 根据这些 API 调用的调用周期而开启或关闭。

请注意,只有在应用调用 setInterruptionFilter(INTERRUPTION_FILTER_ALL) 且预期调用会停用之前由其所有者激活的 AutomaticZenRule 时,此变更才会影响可观察的行为。

Các thay đổi đối với API OpenJDK

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()

新的 SequencedCollection API 可能会影响应用的兼容性 当您在应用的 build 配置中更新 compileSdk 以使用 Android 15(API 级别 35)

  • MutableList.removeFirst()kotlin-stdlib 中的 MutableList.removeLast() 扩展函数

    Java 中的 List 类型会映射到 Kotlin 中的 MutableList 类型。 由于使用 List.removeFirst()List.removeLast() API 已在 Android 15(API 级别 35)中引入,Kotlin 编译器 将函数调用(例如 list.removeFirst())静态解析为 新的 List API,而不是对 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 错误,removeFirst()removeLast() 函数调用可替换为 removeAt(0)removeAt(list.lastIndex)(在 Kotlin 中分别对应)。如果您使用的是 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.getLast();
    }
    

Bảo mật

Android 15 có các thay đổi nhằm tăng cường bảo mật hệ thống để giúp bảo vệ ứng dụng và người dùng khỏi các ứng dụng độc hại.

Chạy hoạt động an toàn ở chế độ nền

Android 15 protects users from malicious apps and gives them more control over their devices by adding changes that prevent malicious background apps from bringing other apps to the foreground, elevating their privileges, and abusing user interaction. Background activity launches have been restricted since Android 10 (API level 29).

Other changes

In addition to the restriction for UID matching, these other changes are also included:

  • Change PendingIntent creators to block background activity launches by default. This helps prevent apps from accidentally creating a PendingIntent that could be abused by malicious actors.
  • Don't bring an app to the foreground unless the PendingIntent sender allows it. This change aims to prevent malicious apps from abusing the ability to start activities in the background. By default, apps are not allowed to bring the task stack to the foreground unless the creator allows background activity launch privileges or the sender has background activity launch privileges.
  • Control how the top activity of a task stack can finish its task. If the top activity finishes a task, Android will go back to whichever task was last active. Moreover, if a non-top activity finishes its task, Android will go back to the home screen; it won't block the finish of this non-top activity.
  • Prevent launching arbitrary activities from other apps into your own task. This change prevents malicious apps from phishing users by creating activities that appear to be from other apps.
  • Block non-visible windows from being considered for background activity launches. This helps prevent malicious apps from abusing background activity launches to display unwanted or malicious content to users.

Ý định an toàn hơn

Android 15 引入了新的安全措施,可让 intent 更加安全且更安全 稳健。这些变更旨在防范潜在的漏洞和 可能会被恶意应用利用的 intent 滥用。有两个主要的 Android 15 中对 intent 安全性的改进:

  • 匹配目标 intent 过滤器:针对特定组件的 intent 准确匹配目标的 intent 过滤器规范。如果您发送 intent 来启动另一个应用的 activity,则目标 intent 组件需要 与接收 activity 声明的 intent 过滤器保持一致。
  • intent 必须具有操作:没有操作的 intent 将不再匹配 任何 intent 过滤器。也就是说,用于启动 activity 或 服务都必须有明确定义的操作。
  • 待处理 intent:待处理 intent 的创建者被视为封装 intent 的发送者,而非待处理 intent 的发送者 意图

Kotlin


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

Java


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

Trải nghiệm người dùng và giao diện người dùng hệ thống

Android 15 có một số thay đổi nhằm tạo ra một tính nhất quán hơn trải nghiệm người dùng trực quan.

Các thay đổi của phần lồng ghép cửa sổ

Android 15 中有两个与窗口边衬区相关的变更:默认强制执行无边框模式;还存在配置变更,例如系统栏的默认配置。

全面实施政策

Theo mặc định, các ứng dụng hiển thị tràn viền trên những thiết bị chạy Android 15 nếu ứng dụng đó nhắm đến Android 15 (API cấp 35).

Một ứng dụng nhắm đến Android 14 và không tràn viền trên Thiết bị chạy Android 15.


Một ứng dụng nhắm đến Android 15 (API cấp 35) và hiển thị tràn viền trên thiết bị chạy Android 15. Ứng dụng này chủ yếu dùng các thành phần Compose Material 3 tự động áp dụng các phần lồng ghép. Màn hình này không chịu ảnh hưởng tiêu cực của Thực thi tràn viền trên Android 15.

Đây là một thay đổi có thể gây lỗi và có thể tác động tiêu cực đến giao diện người dùng của ứng dụng. Chiến lược phát hành đĩa đơn các thay đổi sẽ ảnh hưởng đến các khu vực sau đây trên giao diện người dùng:

  • Thanh điều hướng trên ô điều khiển cử chỉ
    • Minh bạch theo mặc định.
    • Độ lệch dưới cùng bị tắt nên nội dung vẽ phía sau thanh điều hướng của hệ thống trừ phi áp dụng các phần lồng ghép.
    • setNavigationBarColorR.attr#navigationBarColor là và không ảnh hưởng đến thao tác bằng cử chỉ.
    • setNavigationBarContrastEnforcedR.attr#navigationBarContrastEnforced tiếp tục không có ảnh hưởng đến thao tác bằng cử chỉ.
  • Thao tác bằng 3 nút
    • Độ mờ được đặt thành 80% theo mặc định, với màu có thể phù hợp với cửa sổ nền.
    • Đã tắt độ lệch dưới cùng để nội dung vẽ phía sau thanh điều hướng của hệ thống trừ khi áp dụng các phần lồng ghép.
    • setNavigationBarColorR.attr#navigationBarColor là để khớp với nền cửa sổ theo mặc định. Nền cửa sổ phải là màu có thể vẽ để áp dụng giá trị mặc định này. API này là không được dùng nữa nhưng vẫn tiếp tục ảnh hưởng đến cách thao tác bằng 3 nút.
    • setNavigationBarContrastEnforcedR.attr#navigationBarContrastEnforced là true theo mặc định, điều này sẽ thêm Nền mờ 80% trên thao tác bằng 3 nút.
  • Thanh trạng thái
    • Minh bạch theo mặc định.
    • Độ lệch trên cùng bị tắt để nội dung nằm phía sau thanh trạng thái trừ phi phần lồng ghép được áp dụng.
    • setStatusBarColorR.attr#statusBarColor là không dùng nữa và không có hiệu lực trên Android 15.
    • setStatusBarContrastEnforcedR.attr#statusBarContrastEnforced không được dùng nữa nhưng vẫn có một sẽ có hiệu lực trên Android 15.
  • Vết cắt trên màn hình
    • layoutInDisplayCutoutMode cửa sổ không nổi phải là LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS. SHORT_EDGES, NEVERDEFAULT được diễn giải là ALWAYS để người dùng không thấy màu đen thanh do vết cắt trên màn hình tạo ra và hiển thị tràn viền.

Ví dụ sau đây minh hoạ một ứng dụng trước và sau khi nhắm mục tiêu Android 15 (API cấp 35) cũng như trước và sau khi áp dụng các phần lồng ghép.

Một ứng dụng nhắm đến Android 14 và không tràn viền trên Thiết bị chạy Android 15.
Một ứng dụng nhắm đến Android 15 (API cấp 35) và hiển thị tràn viền trên thiết bị chạy Android 15. Tuy nhiên, nhiều yếu tố hiện bị ẩn theo trạng thái thanh, thanh điều hướng bằng 3 nút hoặc vết cắt trên màn hình do Android 15 thực thi tràn viền. Giao diện người dùng ẩn bao gồm Material 2 thanh ứng dụng trên cùng, nút hành động nổi và mục danh sách.
Một ứng dụng nhắm đến Android 15 (API cấp 35) luôn cạnh tranh một thiết bị Android 15 và áp dụng các phần lồng ghép để giao diện người dùng không ẩn.
Những điều cần kiểm tra xem ứng dụng của bạn đã hiển thị tràn viền

Nếu ứng dụng của bạn đã hiện nội dung tràn viền và áp dụng các phần lồng ghép, bạn hầu như không bị ảnh hưởng, ngoại trừ các trường hợp sau đây. Tuy nhiên, ngay cả khi bạn cho rằng thì bạn sẽ không bị ảnh hưởng. Bạn nên kiểm thử ứng dụng của mình.

  • Bạn có một cửa sổ không nổi, chẳng hạn như Activity sử dụng SHORT_EDGES, NEVER hoặc DEFAULT thay vì LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS. Nếu ứng dụng của bạn gặp sự cố khi khởi chạy, điều này có thể là do màn hình chờ. Bạn có thể nâng cấp chính phần phụ thuộc màn hình chờ vào 1.2.0-alpha01 trở lên hoặc thiết lập window.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always.
  • Có thể có những màn hình có lưu lượng truy cập thấp hơn trên giao diện người dùng bị che khuất. Xác minh những thông tin này màn hình ít người truy cập không có giao diện người dùng bị che khuất. Màn hình có lưu lượng truy cập thấp bao gồm:
    • Màn hình giới thiệu hoặc đăng nhập
    • Trang cài đặt
Những điều cần kiểm tra nếu ứng dụng của bạn chưa hiển thị tràn viền

Nếu ứng dụng của bạn chưa phải là nội dung tràn viền thì rất có thể bạn đã bị ảnh hưởng. Ngang bằng ngoài trường hợp các ứng dụng đã hiển thị tràn viền, bạn nên hãy cân nhắc những điều sau:

  • Nếu ứng dụng của bạn dùng Thành phần Material 3 ( androidx.compose.material3) trong Compose, chẳng hạn như TopAppBar, BottomAppBarNavigationBar, các thành phần này có khả năng không bị ảnh hưởng vì chúng tự động xử lý các phần lồng ghép.
  • Nếu ứng dụng của bạn đang dùng Thành phần Material 2 ( androidx.compose.material) trong Compose, các thành phần này không tự động xử lý các phần lồng ghép. Tuy nhiên, bạn có thể truy cập vào các phần lồng ghép và áp dụng chúng theo cách thủ công. Trong androidx.compose.material 1.6.0 và sau đó, dùng tham số windowInsets để áp dụng các phần lồng ghép theo cách thủ công cho BottomAppBar, TopAppBar, BottomNavigationNavigationRail. Tương tự, hãy dùng tham số contentWindowInsets cho Scaffold.
  • Nếu ứng dụng của bạn dùng thành phần hiển thị và Thành phần Material (com.google.android.material), hầu hết Material dựa trên lượt xem Các thành phần như BottomNavigationView, BottomAppBar, NavigationRailView hoặc NavigationView, xử lý các phần lồng ghép và không bắt buộc công việc bổ sung. Tuy nhiên, bạn cần thêm android:fitsSystemWindows="true" nếu dùng AppBarLayout.
  • Đối với các thành phần kết hợp tuỳ chỉnh, hãy áp dụng các phần lồng ghép theo cách thủ công dưới dạng khoảng đệm. Nếu nội dung nằm trong Scaffold, bạn có thể sử dụng các phần lồng ghép bằng cách sử dụng Scaffold giá trị khoảng đệm. Nếu không, hãy áp dụng khoảng đệm bằng một trong WindowInsets.
  • Nếu ứng dụng của bạn đang sử dụng thành phần hiển thị và BottomSheet, SideSheet hoặc thành phần tuỳ chỉnh vùng chứa, áp dụng khoảng đệm bằng cách sử dụng ViewCompat.setOnApplyWindowInsetsListener. Cho RecyclerView, áp dụng khoảng đệm bằng trình nghe này, đồng thời thêm clipToPadding="false".
Những điều cần kiểm tra xem ứng dụng của bạn có phải cung cấp tính năng bảo vệ tuỳ chỉnh trong nền hay không

Nếu ứng dụng của bạn phải cung cấp biện pháp bảo vệ tuỳ chỉnh trong nền cho tính năng thao tác bằng 3 nút hoặc thanh trạng thái, ứng dụng nên đặt một thành phần kết hợp hoặc khung hiển thị phía sau thanh hệ thống sử dụng WindowInsets.Type#tappableElement() để tạo nút 3 chiều cao của thanh điều hướng hoặc WindowInsets.Type#statusBars.

Tài nguyên bổ sung tràn viền

Xem Chế độ xem Edge to EdgeEdge to Edge Compose để biết thêm các cân nhắc khi áp dụng các phần lồng ghép.

API không dùng nữa

Các API sau đây hiện không được dùng nữa:

稳定配置

If your app targets Android 15 (API level 35) or higher, Configuration no longer excludes the system bars. If you use the screen size in the Configuration class for layout calculation, you should replace it with better alternatives like an appropriate ViewGroup, WindowInsets, or WindowMetricsCalculator depending on your need.

Configuration has been available since API 1. It is typically obtained from Activity.onConfigurationChanged. It provides information like window density, orientation, and sizes. One important characteristic about the window sizes returned from Configuration is that it previously excluded the system bars.

The configuration size is typically used for resource selection, such as /res/layout-h500dp, and this is still a valid use case. However, using it for layout calculation has always been discouraged. If you do so, you should move away from it now. You should replace the use of Configuration with something more suitable depending on your use case.

If you use it to calculate the layout, use an appropriate ViewGroup, such as CoordinatorLayout or ConstraintLayout. If you use it to determine the height of the system navbar, use WindowInsets. If you want to know the current size of your app window, use computeCurrentWindowMetrics.

The following list describes the fields affected by this change:

Giá trị mặc định của thuộc tính billingTextHeight là true

对于以 Android 15 为目标平台的应用,elegantTextHeight TextView 属性默认变为 true,将默认使用的紧凑字体替换为一些具有较大垂直指标的脚本,并且这种字体更易于阅读。紧凑字体的引入是为了防止破坏布局;Android 13(API 级别 33)允许文本布局利用 fallbackLineSpacing 属性拉伸垂直高度,以防止许多此类破坏。

在 Android 15 中,紧凑字体仍保留在系统中,因此您的应用可以将 elegantTextHeight 设置为 false,以获得与之前相同的行为,但即将在未来版本中提供支持。因此,如果您的应用支持以下文字:阿拉伯语、老挝语、缅甸、泰米尔语、古吉拉特语、卡纳达语、马拉雅拉姆语、奥里亚语、泰卢固语或泰语,请将 elegantTextHeight 设置为 true,以测试应用。

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

Thay đổi chiều rộng của TextView cho các hình dạng chữ cái phức tạp

In previous versions of Android, some cursive fonts or languages that have complex shaping might draw the letters in the previous or next character's area. In some cases, such letters were clipped at the beginning or ending position. Starting in Android 15, a TextView allocates width for drawing enough space for such letters and allows apps to request extra paddings to the left to prevent clipping.

Because this change affects how a TextView decides the width, TextView allocates more width by default if the app targets Android 15 (API level 35) or higher. You can enable or disable this behavior by calling the setUseBoundsForWidth API on TextView.

Because adding left padding might cause a misalignment for existing layouts, the padding is not added by default even for apps that target Android 15 or higher. However, you can add extra padding to preventing clipping by calling setShiftDrawingOffsetForStartOverhang.

The following examples show how these changes can improve text layout for some fonts and languages.

Standard layout for English text in a cursive font. Some of the letters are clipped. Here is the corresponding XML:

<TextView
    android:fontFamily="cursive"
    android:text="java" />
Layout for the same English text with additional width and padding. Here is the corresponding XML:

<TextView
    android:fontFamily="cursive"
    android:text="java"
    android:useBoundsForWidth="true"
    android:shiftDrawingOffsetForStartOverhang="true" />
Standard layout for Thai text. Some of the letters are clipped. Here is the corresponding XML:

<TextView
    android:text="คอมพิวเตอร์" />
Layout for the same Thai text with additional width and padding. Here is the corresponding XML:

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

Chiều cao dòng mặc định có thể nhận biết ngôn ngữ cho EditText

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

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

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

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

如果需要,您的应用可以通过将 useLocalePreferredLineHeightForMinimum 属性设置为 false 来恢复之前的行为,并且可以通过 Kotlin 和 Java 中的 setMinimumFontMetrics API 设置自定义最小行业指标。

Máy ảnh và nội dung nghe nhìn

Android 15 thực hiện các thay đổi sau đối với hành vi của máy ảnh và nội dung nghe nhìn đối với ứng dụng nhắm đến Android 15 trở lên.

Quy định hạn chế đối với việc yêu cầu quyền phát âm thanh

以 Android 15 为目标平台的应用必须是顶级应用或运行前台服务,才能请求音频焦点。如果应用在不符合其中任何一项要求时尝试请求焦点,该调用将返回 AUDIOFOCUS_REQUEST_FAILED

您可以参阅管理音频焦点,详细了解音频焦点。

Các quy tắc hạn chế mới cập nhật đối với yếu tố ngoài SDK

Android 15 包含更新后的受限非 SDK 列表 基于与 Android 开发者之间的协作以及最新的 内部测试在限制使用非 SDK 接口之前,我们会尽可能确保有可用的公开替代方案。

如果您的应用并非以 Android 15 为目标平台,其中一些变更可能不会立即对您产生影响。不过,虽然您的应用可以访问一些非 SDK 接口(具体取决于应用的目标 API 级别),但如果您使用任何非 SDK 方法或字段,应用无法运行的风险始终会很高。

如果您不确定自己的应用是否使用了非 SDK 接口,可以 测试您的应用,找出答案。如果您的应用依赖于非 SDK 接口,则应开始计划迁移到 SDK 替代方案。 尽管如此,我们了解到,某些应用具有 非 SDK 接口。如果您无法为应用中的某项功能找到使用非 SDK 接口的替代方案,则应请求新的公共 API

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