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:
- 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ọistopSelf()
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.) - Đả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ờ). - 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. - 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:
- 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
(自 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
限制在应用拥有 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(setInterruptionFilter
、setNotificationPolicy
)会导致创建或更新隐式 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: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()
。
新的 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 中的其他方法冲突
向现有类型中添加了新方法,例如,
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.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 aPendingIntent
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).
Đâ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.
setNavigationBarColor
vàR.attr#navigationBarColor
là và không ảnh hưởng đến thao tác bằng cử chỉ.setNavigationBarContrastEnforced
vàR.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.
setNavigationBarColor
vàR.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.setNavigationBarContrastEnforced
vàR.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.
setStatusBarColor
vàR.attr#statusBarColor
là không dùng nữa và không có hiệu lực trên Android 15.setStatusBarContrastEnforced
vàR.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
,NEVER
vàDEFAULT
đượ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.
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ụngSHORT_EDGES
,NEVER
hoặcDEFAULT
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ậpwindow.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
,BottomAppBar
vàNavigationBar
, 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 choBottomAppBar
,TopAppBar
,BottomNavigation
vàNavigationRail
. Tương tự, hãy dùng tham sốcontentWindowInsets
choScaffold
. - 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ặcNavigationView
, 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êmandroid:fitsSystemWindows="true"
nếu dùngAppBarLayout
. - Đố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ụngScaffold
giá trị khoảng đệm. Nếu không, hãy áp dụng khoảng đệm bằng một trongWindowInsets
. - 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ụngViewCompat.setOnApplyWindowInsetsListener
. ChoRecyclerView
, áp dụng khoảng đệm bằng trình nghe này, đồng thời thêmclipToPadding="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 Edge và Edge 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:
R.attr#enforceStatusBarContrast
R.attr#navigationBarColor
R.attr#navigationBarDividerColor
R.attr#statusBarColor
Window#getNavigationBarColor
Window#getNavigationBarDividerColor
Window#getStatusBarColor
Window#isStatusBarContrastEnforced
Window#setDecorFitsSystemWindows
Window#setNavigationBarColor
Window#setNavigationBarDividerColor
Window#setStatusBarColor
Window#setStatusBarContrastEnforced
稳定配置
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:
Configuration.screenWidthDp
andscreenHeightDp
sizes no longer exclude the system bars.Configuration.smallestScreenWidthDp
is indirectly affected by changes toscreenWidthDp
andscreenHeightDp
.Configuration.orientation
is indirectly affected by changes toscreenWidthDp
andscreenHeightDp
on close-to-square devices.Display.getSize(Point)
is indirectly affected by the changes inConfiguration
. This was deprecated beginning in API level 30.Display.getMetrics()
has already worked like this since API level 33.
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
,以测试应用。
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.
Chiều cao dòng mặc định có thể nhận biết ngôn ngữ cho EditText
在以前的 Android 版本中,文本布局拉伸了文本的高度,使其适应与当前语言区域匹配的字体的行高。例如,如果内容是日语,由于日语字体的行高比拉丁字体的行高略大,因此文本的高度就略大了。不过,尽管行高存在这些差异,但无论使用何种语言区域,EditText
元素的大小都是一致的,如下图所示:
对于以 Android 15 为目标平台的应用,系统现在会为 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 接口的限制。