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

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

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

使用者體驗和系統 UI

Android 16 (API 級別 36) 包含下列變更,旨在打造更一致、直覺的使用者體驗。

無邊框設計停用選項即將移除

Android 15 强制执行全屏显示,但您的应用可以通过将 R.attr#windowOptOutEdgeToEdgeEnforcement 设置为 true 来选择停用此功能。对于以 Android 16(API 级别 36)为目标平台的应用,R.attr#windowOptOutEdgeToEdgeEnforcement 已被废弃并停用,并且您的应用无法选择不采用从边缘到边缘的布局。

  • 如果您的应用以 Android 16(API 级别 36)为目标平台,并且在 Android 15 设备上运行,则 R.attr#windowOptOutEdgeToEdgeEnforcement 会继续正常运行。
  • 如果您的应用以 Android 16(API 级别 36)为目标平台,并且在 Android 16 设备上运行,则 R.attr#windowOptOutEdgeToEdgeEnforcement 会被停用。

如需在 Android 16 中进行测试,请确保您的应用支持无边框设计,并移除所有 R.attr#windowOptOutEdgeToEdgeEnforcement 用法,以便您的应用在 Android 15 设备上也能支持无边框设计。如需支持从边缘到边缘的显示,请参阅 ComposeViews 指南。

如要使用預測返回手勢,必須遷移或選擇停用

对于以 Android 16(API 级别 36)或更高版本为目标平台且在搭载 Android 16 或更高版本的设备上运行的应用,预测性返回系统动画(返回主屏幕、跨任务和跨 activity)默认处于启用状态。此外,系统不再调用 onBackPressed,也不再调度 KeyEvent.KEYCODE_BACK

如果您的应用会拦截返回事件,但您尚未迁移到预测性返回,请更新应用以使用受支持的返回导航 API,或者通过在应用的 AndroidManifest.xml 文件的 <application><activity> 标记中将 android:enableOnBackInvokedCallback 属性设置为 false 来暂时选择停用。

预测性返回首页动画。
预测性跨 activity 动画。
预测性跨任务动画。

已淘汰並停用 Elegant 字型 API

指定 Android 15 (API 級別 35) 的應用程式會預設將 elegantTextHeight TextView 屬性設為 true,以更易於閱讀的字型取代精簡字型。您可以將 elegantTextHeight 屬性設為 false,藉此覆寫這項設定。

Android 16 會淘汰 elegantTextHeight 屬性,應用程式指定 Android 16 後,系統就會忽略該屬性。這些 API 控制的「UI 字型」即將停用,因此請調整所有版面配置,確保阿拉伯文、寮文、緬甸文、泰米爾文、古吉拉特文、卡納達文、馬拉雅拉姆文、奧里亞文、泰盧固文或泰文的文字顯示方式一致,且日後不會出錯。

針對指定 Android 14 (API 級別 34) 以下版本的應用程式,或指定 Android 15 (API 級別 35) 但將 elegantTextHeight 屬性設為 false,藉此覆寫預設行為的應用程式,
elegantTextHeight 行為如下:
如果應用程式指定 Android 16 (API 級別 36),或指定 Android 15 (API 級別 35) 但未將 elegantTextHeight 屬性設為 false 來覆寫預設值,則會採用
elegantTextHeight 行為。

核心功能

Android 16 (API 級別 36) 包含下列變更,可修改或擴充 Android 系統的各種核心功能。

固定費率工作排程最佳化

在以 Android 16 为目标平台之前,如果 scheduleAtFixedRate 因不在有效的进程生命周期内而错过了任务执行,则当应用返回到有效的生命周期时,所有错过的执行会立即执行。

以 Android 16 为目标平台时,当应用返回到有效的生命周期时,系统会立即执行最多 1 次未执行的 scheduleAtFixedRate 执行。此行为变更预计会提升应用性能。在您的应用中测试此行为,检查您的应用是否受到影响。您还可以使用应用兼容性框架并启用 STPE_SKIP_MULTIPLE_MISSED_PERIODIC_TASKS 兼容性标志进行测试。

裝置板型規格

在大型螢幕裝置上顯示時,Android 16 (API 級別 36) 會對應用程式進行下列變更。

自動調整式版面配置

Android 應用程式現在可在各種裝置上執行 (例如手機、平板電腦、折疊式裝置、桌機、車輛和電視),並支援大螢幕上的視窗模式 (例如分割畫面和電腦視窗),因此開發人員應建構可配合任何螢幕和視窗大小調整的 Android 應用程式,無論裝置螢幕方向為何。在現今多裝置的世界中,限制螢幕方向和大小調整等範例過於嚴苛。

忽略螢幕方向、是否可調整大小和顯示比例限制

如果應用程式指定 Android 16 (API 級別 36),螢幕方向、大小調整和螢幕比例限制將不再適用於最小寬度 >= 600 dp 的螢幕。無論顯示比例或使用者偏好的螢幕方向為何,應用程式都會填滿整個顯示視窗,且不會出現側邊黑邊。

這項異動會導入新的標準平台行為。Android 正在朝向某種模式發展,預計應用程式將能適應各種螢幕方向、顯示大小和顯示比例。如果應用程式設有固定螢幕方向或大小調整限制,就無法適應各種裝置。讓應用程式具備適應性,提供最佳使用者體驗。

您也可以使用應用程式相容性架構,並啟用 UNIVERSAL_RESIZABLE_BY_DEFAULT 相容性標記,測試這項行為。

常見的破壞性變更

忽略方向、可調整大小和長寬比限制,可能會影響應用程式在某些裝置上的 UI,尤其是專為鎖定直向的小版面配置設計的元素,例如版面配置遭到延展,以及動畫和元件超出螢幕範圍等問題。如果對顯示比例或螢幕方向做出任何假設,可能會導致應用程式出現視覺問題。請參閱這篇文章,進一步瞭解如何避免這類問題,並改善應用程式的適應性行為。

允許裝置旋轉會導致更多活動重建,如果未妥善保留,可能會導致使用者狀態遺失。如要瞭解如何正確儲存 UI 狀態,請參閱「儲存 UI 狀態」。

導入作業詳細資料

在全螢幕和多視窗模式下,大型螢幕裝置會忽略下列資訊清單屬性和執行階段 API:

系統會忽略 screenOrientationsetRequestedOrientation()getRequestedOrientation() 的下列值:

  • portrait
  • reversePortrait
  • sensorPortrait
  • userPortrait
  • landscape
  • reverseLandscape
  • sensorLandscape
  • userLandscape

就螢幕大小調整功能而言,android:resizeableActivity="false"android:minAspectRatioandroid:maxAspectRatio 不會產生任何影響。

如果應用程式以 Android 16 (API 級別 36) 為目標,系統預設會忽略大螢幕上的應用程式方向、大小調整和長寬比限制,但每個尚未完全準備就緒的應用程式都可以暫時選擇停用這項行為 (這會導致應用程式進入相容性模式)。

例外狀況

在下列情況中,Android 16 的螢幕方向、大小調整和長寬比限制不適用:

  • 遊戲 (根據 android:appCategory 旗標)
  • 使用者在裝置的顯示比例設定中,明確選擇採用應用程式的預設行為
  • 小於 sw600dp 的螢幕

暫時退出

如要停用特定活動,請宣告 PROPERTY_COMPAT_ALLOW_RESTRICTED_RESIZABILITY 資訊清單屬性:

<activity ...>
  <property android:name="android.window.PROPERTY_COMPAT_ALLOW_RESTRICTED_RESIZABILITY" android:value="true" />
  ...
</activity>

如果應用程式有太多部分尚未支援 Android 16,您可以在應用程式層級套用相同屬性,完全停用這項功能:

<application ...>
  <property android:name="android.window.PROPERTY_COMPAT_ALLOW_RESTRICTED_RESIZABILITY" android:value="true" />
</application>

健康與健身

Android 16 (API 級別 36) 包含下列與健康和健身資料相關的變更。

健康與健身權限

对于以 Android 16(API 级别 36)或更高版本为目标平台的应用,BODY_SENSORS 权限使用 android.permissions.health 下更精细的权限,健康数据共享也使用这些权限。自 Android 16 起,凡是以前需要具有 BODY_SENSORSBODY_SENSORS_BACKGROUND 权限的 API,现在都需要获取相应的 android.permissions.health 权限。这会影响以下数据类型、API 和前台服务类型:

如果您的应用使用这些 API,则应请求相应的精细权限:

这些权限与用于保护对 Health Connect(Android 健康、健身和身心状态数据存储区)中读取数据的访问权限相同。

移动应用

迁移到使用 READ_HEART_RATE 和其他精细权限的移动应用还必须声明 activity 以显示应用的隐私权政策。此要求与健康数据共享的要求相同。

連線能力

Android 16 (API 級別 36) 包含藍牙堆疊的下列變更,可提升與周邊裝置的連線能力。

處理債券遺失和加密變更的新意圖

除了改善連結遺失處理功能,Android 16 也推出了 2 個新的意圖,讓應用程式更能察覺連結遺失和加密變更。

鎖定 Android 16 的應用程式現在可以:

  • 在偵測到遠端連結中斷時接收 ACTION_KEY_MISSING 意圖,以便提供更有用的使用者意見回饋,並採取適當行動。
  • 每當連結的加密狀態變更時,就會收到 ACTION_ENCRYPTION_CHANGE 意圖。包括加密狀態變更、加密演算法變更和加密金鑰大小變更。如果應用程式稍後收到 ACTION_ENCRYPTION_CHANGE 意圖,且連結已成功加密,則必須考慮恢復連結。

因應不同原始設備製造商 (OEM) 實作

雖然 Android 16 推出了這些新意圖,但不同裝置製造商 (OEM) 的實作和廣播方式可能有所不同。為確保應用程式可在所有裝置上提供一致且可靠的體驗,開發人員應設計連結中斷處理機制,以便妥善因應這些潛在的變化。

我們建議您採用下列應用程式行為:

  • 如果廣播 ACTION_KEY_MISSING 意圖:

    系統會中斷 ACL (非同步無連結) 連結,但會保留裝置的連結資訊 (如這裡所述)。

    應用程式應使用這個意圖做為偵測連結中斷的主要信號,並在啟動裝置忘記或重新配對功能前,引導使用者確認遠端裝置是否在範圍內。

    如果裝置在收到 ACTION_KEY_MISSING 後中斷連線,應用程式應謹慎重新連線,因為裝置可能已不再與系統綁定。

  • 如果 ACTION_KEY_MISSING 意圖未廣播:

    ACL 連結會保持連線狀態,系統會移除裝置的連結資訊,這與 Android 15 的行為相同。

    在這種情況下,應用程式應繼續使用與先前 Android 版本相同的現有連結失效處理機制,以偵測及管理連結失效事件。

移除藍牙配對的新方式

所有以 Android 16 為目標版本的應用程式,現在都能使用 CompanionDeviceManager 中的公開 API 解除配對藍牙裝置。如果隨附裝置是以 CDM 關聯方式管理,應用程式可以在關聯裝置上使用新的 removeBond(int) API 觸發藍牙連結移除作業。應用程式可以監控連結狀態變更,方法是監聽藍牙裝置廣播事件 ACTION_BOND_STATE_CHANGED

安全性

Android 16 (API 級別 36) 包含下列安全性異動。

MediaStore 版本鎖定

針對指定 Android 16 以上版本的應用程式,MediaStore#getVersion() 現已成為每個應用程式的專屬值。這麼做可移除版本字串中的識別屬性,以免遭到濫用,並防止用於指紋辨識技術。應用程式不應對此版本的格式做出任何假設。應用程式在使用此 API 時應已處理版本變更,且在大多數情況下,不必變更目前的行為,除非開發人員嘗試推斷超出此 API 預期範圍的其他資訊。

更安全的意圖

“更安全的 intent”功能是一项多阶段安全计划,旨在提高 Android 的 intent 解析机制的安全性。目标是在 intent 处理期间添加检查,并过滤不符合特定条件的 intent,从而保护应用免受恶意操作的侵害。

Android 15 中,该功能侧重于发送应用,现在在 Android 16 中,控制权转移到了接收应用,允许开发者使用其应用清单选择加入严格的 intent 解析。

我们正在实施两项关键变更:

  1. 显式 intent 必须与目标组件的 intent 过滤器相匹配:如果 intent 显式定位到某个组件,则应与该组件的 intent 过滤器相匹配。

  2. 没有操作的 intent 无法匹配任何 intent 过滤器:未指定操作的 intent 不应解析为任何 intent 过滤器。

这些变更仅在涉及多个应用时适用,不会影响单个应用内的 intent 处理。

影响

选择启用性质意味着,开发者必须在应用清单中明确启用它,才能使其生效。 因此,此功能的影响将仅限于以下应用:

  • 了解“更安全的 intent”功能及其优势。
  • 主动选择在应用中采用更严格的 intent 处理实践。

这种选择性采用的方法可最大限度地降低破坏可能依赖于当前不太安全的 intent 解析行为的现有应用的风险。

虽然在 Android 16 中,初始影响可能有限,但“更安全的 intent”计划的路线图显示,未来 Android 版本的影响范围会更广。我们计划最终将严格的意图解析设为默认行为。

“更安全的 intent”功能可让恶意应用更难利用 intent 解析机制中的漏洞,从而有望显著提升 Android 生态系统的安全性。

不过,向选择退出和强制执行的过渡必须谨慎管理,以解决现有应用的潜在兼容性问题。

实现

开发者需要在应用清单中使用 intentMatchingFlags 属性明确启用更严格的 intent 匹配。 以下示例展示了如何为整个应用选择启用该功能,但在接收器上停用/选择停用该功能:

<application android:intentMatchingFlags="enforceIntentFilter">
    <receiver android:name=".MyBroadcastReceiver" android:exported="true" android:intentMatchingFlags="none">
        <intent-filter>
            <action android:name="com.example.MY_CUSTOM_ACTION" />
        </intent-filter>
        <intent-filter>
            <action android:name="com.example.MY_ANOTHER_CUSTOM_ACTION" />
        </intent-filter>
    </receiver>
</application>

有关支持的标志的更多信息:

标志名称 说明
enforceIntentFilter 对传入的 intent 强制执行更严格的匹配
none 停用针对传入 intent 的所有特殊匹配规则。指定多个标志时,系统会优先考虑“无”标志,以解决值冲突问题
allowNullAction 放宽了匹配规则,允许匹配没有操作的 intent。此标志与“enforceIntentFilter”结合使用可实现特定行为

测试和调试

在强制执行处于有效状态时,如果 intent 调用方已正确填充 intent,应用应能正常运行。 不过,被屏蔽的 intent 会触发警告日志消息(例如 "Intent does not match component's intent filter:""Access blocked:"),并带有标记 "PackageManager."。这表示存在可能会影响应用的潜在问题,需要引起注意。

Logcat 过滤条件:

tag=:PackageManager & (message:"Intent does not match component's intent filter:" | message: "Access blocked:")

GPU 系統呼叫篩選

为了加固 Mali GPU 表面,在生产 build 中,已废弃或仅用于 GPU 开发的 Mali GPU IOCTL 已被屏蔽。此外,用于 GPU 性能分析的 IOCTL 已限制为 shell 进程或可调试的应用。如需详细了解平台级政策,请参阅 SAC 更新。

此更改适用于使用 Mali GPU 的 Pixel 设备(Pixel 6-9)。Arm 已在其 r54p2 版本Documentation/ioctl-categories.rst 中提供了 IOCTL 的官方分类。此列表将在未来的驱动程序版本中继续维护。

此项变更不会影响受支持的图形 API(包括 Vulkan 和 OpenGL),预计也不会影响开发者或现有应用。 Streamline Performance Analyzer 和 Android GPU 检查器等 GPU 性能剖析工具不会受到影响。

测试

如果您看到类似如下所示的 SELinux 拒绝,则说明您的应用可能受到了此变更的影响:

06-30 10:47:18.617 20360 20360 W roidJUnitRunner: type=1400 audit(0.0:85): avc:  denied  { ioctl }
for  path="/dev/mali0" dev="tmpfs" ino=1188 ioctlcmd=0x8023
scontext=u:r:untrusted_app_25:s0:c512,c768 tcontext=u:object_r:gpu_device:s0 tclass=chr_file
permissive=0 app=com.google.android.selinux.pts

如果您的应用需要使用被屏蔽的 IOCTL,请提交 bug 并将其分配给 android-partner-security@google.com。

常见问题解答

  1. 此政策变更是否适用于所有原始设备制造商 (OEM)? 此变更将采用选择启用模式,但任何想要使用此强化方法的 OEM 都可以使用。如需了解如何实现此变更,请参阅实现文档。

  2. 是否必须在 OEM 代码库中进行更改才能实现此功能,还是默认随新的 AOSP 版本提供? 平台级变更将默认随新的 AOSP 版本一起发布。如果供应商想要应用此变更,可以在其代码库中选择启用此变更。

  3. SoC 是否负责使 IOCTL 列表保持最新状态?例如,如果我的设备使用 ARM Mali GPU,我是否需要就任何更改与 ARM 联系? 各个 SoC 必须在驱动程序发布后根据设备更新其 IOCTL 列表。 例如,ARM 会在驱动程序更新时更新其已发布的 IOCTL 列表。 不过,OEM 应确保在 SEPolicy 中纳入这些更新,并根据需要将任何选定的自定义 IOCTL 添加到列表中。

  4. 此变更是否会自动应用于所有在售 Pixel 设备,还是需要用户执行操作来切换某些设置才能应用此变更? 此变更适用于所有使用 Mali GPU 的 Pixel 在售设备(Pixel 6-9)。用户无需采取任何行动即可应用此变更。

  5. 使用此政策会影响内核驱动程序的性能吗? 我们使用 GFXBench 在 Mali GPU 上测试了此政策,未发现 GPU 性能有任何可衡量的变化。

  6. IOCTL 列表是否需要与当前的用户空间和内核驱动程序版本保持一致? 是的,允许的 IOCTL 列表必须与用户空间和内核驱动程序支持的 IOCTL 同步。如果用户空间或内核驱动程序中的 IOCTL 发生更新,则必须更新 SEPolicy IOCTL 列表以保持一致。

  7. ARM 已将 IOCTL 分类为“受限”/“检测”,但我们希望在生产用例中使用其中一些 IOCTL,并拒绝其他 IOCTL。 各个 OEM/SoC 负责根据其用户空间 Mali 库的配置来决定如何对其使用的 IOCTL 进行分类。ARM 的列表可用于帮助确定这些值,但每个 OEM/SoC 的使用情形可能有所不同。

隱私權

Android 16 (API 級別 36) 包含下列隱私權異動。

區域網路權限

只要應用程式具備 INTERNET 權限,就能存取 LAN 上的裝置。 這項功能可讓應用程式輕鬆連線至本機裝置,但也會產生隱私權影響,例如形成使用者指紋,以及成為位置資訊的 Proxy。

「區域網路保護」專案的目標是透過新的執行階段權限,限制區域網路的存取權,進而保護使用者隱私權。

發布計畫

這項異動將在 2025 年第 2 季和 2026 年第 2 季之間推出。開發人員務必遵循 25Q2 的這項指引,並分享意見回饋,因為這些保護措施將在日後的 Android 版本中強制執行。此外,他們也需要按照下列指引更新依附於隱含本機網路存取的案例,並準備好因應使用者拒絕或撤銷新權限的情況。

影響

現階段,LNP 是選擇加入的功能,因此只有選擇加入的應用程式會受到影響。在選擇加入階段,應用程式開發人員的目標是瞭解應用程式的哪些部分依附於隱含的區域網路存取權,以便為下一個版本準備權限防護措施。

如果應用程式使用下列方式存取使用者的本機網路,就會受到影響:

  • 直接或透過程式庫使用本機網路位址的原始通訊端 (例如 mDNS 或 SSDP 服務探索通訊協定)
  • 使用可存取本機網路的架構層級類別 (例如 NsdManager)

傳送至傳送自區域網路位址的流量,需要區域網路存取權。下表列出一些常見案例:

應用程式低層級網路作業 必須授予區域網路權限
建立外送 TCP 連線
接受傳入的 TCP 連線
傳送 UDP 單點傳播、多點傳播、廣播
接收傳入的 UDP 單點傳播、多點傳播、廣播

這些限制是在網路堆疊深處實作,因此適用於所有網路 API。包括以原生或受管理程式碼建立的通訊端、Cronet 和 OkHttp 等網路程式庫,以及在這些程式庫上實作的任何 API。如要解析區域網路上的服務 (即含有 .local 後置字元的服務),必須具備區域網路權限。

上述規則的例外狀況:

  • 如果裝置的 DNS 伺服器位於區域網路上,則往返該伺服器的流量 (通訊埠 53) 不需要區域網路存取權。
  • 如果應用程式使用 Output Switcher 做為應用程式內選取器,則不需要區域網路權限 (更多指引將於 2025 年第 4 季發布)。

開發人員指引 (選擇加入)

如要啟用區域網路限制,請按照下列步驟操作:

  1. 將裝置刷機為 25Q2 Beta 3 以上版本。
  2. 安裝要測試的應用程式。
  3. 在 adb 中切換 Appcompat 標記:

    adb shell am compat enable RESTRICT_LOCAL_NETWORK <package_name>
    
  4. 重新啟動裝置

現在應用程式的區域網路存取權受到限制,任何存取區域網路的嘗試都會導致通訊端錯誤。如果您使用的 API 會在應用程式程序外部執行本機網路作業 (例如 NsdManager),在選擇加入階段不會受到影響。

如要還原存取權,您必須授予應用程式 NEARBY_WIFI_DEVICES 權限。

  1. 確認應用程式在資訊清單中宣告了 NEARBY_WIFI_DEVICES 權限。
  2. 依序前往「設定」>「應用程式」>「[應用程式名稱]」>「權限」>「鄰近裝置」>「允許」

現在應用程式應該已恢復區域網路存取權,所有情境也應與應用程式加入測試前一樣運作。

區域網路保護措施開始強制執行後,應用程式網路流量會受到以下影響。

權限 傳出 LAN 要求 傳出/傳入網際網路要求 傳入 LAN 要求
已授權 Works Works Works
未授予 凸槌影片 Works 凸槌影片

使用下列指令切換關閉 App-Compat 旗標

adb shell am compat disable RESTRICT_LOCAL_NETWORK <package_name>

錯誤

每當呼叫插座叫用 send 或 send 變數至本機網路位址時,系統就會將因這些限制而產生的錯誤傳回呼叫插座。

錯誤示例:

sendto failed: EPERM (Operation not permitted)

sendto failed: ECONNABORTED (Operation not permitted)

區域網路定義

本專案中的本機網路是指使用具備廣播功能的網路介面 (例如 Wi-Fi 或乙太網路) 的 IP 網路,但不包括行動網路 (WWAN) 或 VPN 連線。

下列為本機網路:

IPv4:

  • 169.254.0.0/16 // 連結本機
  • 100.64.0.0/10 // CGNAT
  • 10.0.0.0/8 // RFC1918
  • 172.16.0.0/12 // RFC1918
  • 192.168.0.0/16 // RFC1918

IPv6:

  • 連結本機
  • 直接連線的路徑
  • Thread 等存根網路
  • 多個子網路 (待定)

此外,多點播送位址 (224.0.0.0/4、ff00::/8) 和 IPv4 廣播位址 (255.255.255.255) 都屬於本機網路位址。

應用程式擁有的相片

当面向 SDK 36 或更高版本的应用在搭载 Android 16 或更高版本的设备上提示用户授予照片和视频权限时,如果用户选择限制对所选媒体的访问权限,则会在照片选择器中看到该应用拥有的所有照片。用户可以取消选择任何这些预选项,这会撤消该应用对这些照片和视频的访问权限。