行為變更:指定 Android 12 為目標的應用程式

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

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

使用者體驗

自訂通知

Android 12 會變更完全自訂通知的外觀和行為。在過去,自訂通知可以使用整個通知區域,並自行提供版面配置和樣式。這會導致反模式,可能會讓使用者感到困惑,或在不同裝置上造成版面配置相容性問題。

如果是指定 Android 12 為目標版本的應用程式,附帶自訂內容檢視畫面的通知將不再使用整個通知區域,系統會改為套用標準範本。這個範本可確保自訂通知與所有狀態的其他通知採用相同的裝飾設定,例如通知的圖示和展開操作功能 (在收合狀態),以及通知的圖示、應用程式名稱和展開操作功能 (在展開狀態)。這個行為與 Notification.DecoratedCustomViewStyle 的行為幾乎相同。

如此一來,Android 12 就能讓所有通知在視覺上保持一致,方便使用者掃描,並提供可供使用者探索的熟悉通知展開功能。

下圖顯示標準範本中的自訂通知:

以下範例說明自訂通知在收合和展開狀態下的顯示方式:

Android 12 的變更會影響定義 Notification.Style 自訂子類別,或使用 Notification.Builder 方法 setCustomContentView(RemoteViews)setCustomBigContentView(RemoteViews)setCustomHeadsUpContentView(RemoteViews) 的應用程式。

如果您的應用程式使用完全自訂的通知,建議您盡快使用新範本進行測試。

  1. 啟用自訂通知變更:

    1. 將應用程式的 targetSdkVersion 變更為 S,即可啟用新行為。
    2. 重新編譯。
    3. 在搭載 Android 12 的裝置或模擬器上安裝應用程式。
  2. 測試所有使用自訂檢視畫面的通知,確保通知在通知面板中顯示的樣貌符合您的預期。測試時,請將以下考量因素納入考量,並進行必要調整:

    • 自訂檢視畫面的尺寸已變更。一般來說,自訂通知可用的高度會比之前少。在收合狀態下,自訂內容的高度上限已從 106dp 降至 48dp。此外,橫向空間也較少。

    • 指定 Android 12 為目標的應用程式可展開所有通知。一般來說,這表示如果您使用 setCustomContentView,也應使用 setBigCustomContentView,確保收合和展開狀態一致。

    • 為確保「看路提醒」狀態符合預期,別忘了將通知管道的重要性提高為「高」(畫面上的彈出式視窗)。

針對以 Android 12 以上版本為目標版本的應用程式,系統會對 Android 應用程式連結的驗證方式做出幾項變更。這些變更可提供更穩定的應用程式連結體驗,讓應用程式開發人員和使用者手上有更多掌控權。

如果您依賴 Android 應用程式連結驗證功能,在應用程式中開啟網頁連結,請確認您在新增意圖篩選器時,使用正確的格式進行 Android 應用程式連結驗證。請特別確認這些意圖篩選器包含 BROWSABLE 類別,且支援 https 配置。

您也可以手動驗證應用程式連結,測試宣告的可靠性。

改善子母畫面行為

Android 12 針對子母畫面 (PiP) 模式推出行為改善功能,並建議針對手勢操作和元素導覽的轉場動畫進行外觀改善。

詳情請參閱「子母畫面功能改善」。

吐司重新設計

在 Android 12 中,Toast 檢視畫面已重新設計。浮動式訊息現在以兩行文字為限,且會在文字旁顯示應用程式圖示。

圖中的 Android 裝置顯示彈出的浮動式訊息,而訊息內的應用程式圖示旁顯示「正在傳送郵件」

詳情請參閱「Toast 總覽」。

安全性和隱私權

大概位置

在搭載 Android 12 以上版本的裝置上,使用者可以要求應用程式提供大概位置精確度

WebView 中的新式 SameSite Cookie

Android 的 WebView 元件以 Chromium 為基礎,該檔案是採用 Google Chrome 瀏覽器運作的開放原始碼專案。Chromium 推出了調整第三方 Cookie 的處理方式,以提升安全性和隱私權,並讓使用者能進一步掌握相關資訊。自 Android 12 起,當應用程式指定 Android 12 (API 級別 31) 以上版本時,這些變更也會納入 WebView

Cookie 的 SameSite 屬性可控制 Cookie 是否可透過任何要求傳送,或是僅限同網站要求傳送。下列隱私權保護變更可改善第三方 Cookie 的預設處理方式,並協助防範不慎的跨網站分享行為:

  • 不含 SameSite 屬性的 Cookie 會視為 SameSite=Lax
  • 含有 SameSite=None 的 Cookie 也必須指定 Secure 屬性,表示 Cookie 需要安全的環境,且應透過 HTTPS 傳送。
  • 網站的 HTTP 和 HTTPS 版本之間的連結現在會視為跨網站要求,因此除非 Cookie 適當標示為 SameSite=None; Secure,否則不會傳送 Cookie。

對於開發人員而言,一般建議是找出重要使用者流程中的跨網站 Cookie 依附元件,並確保在需要時,明確設定 SameSite 屬性,並使用適當的值。您必須明確指定可在不同網站或從 HTTP 轉換為 HTTPS 的相同網站導覽中運作的 Cookie。

如需網頁開發人員這些變更的完整指南,請參閱 SameSite Cookie 說明Schemeful SameSite

測試應用程式中的 SameSite 行為

如果您的應用程式使用 WebView,或是您管理的是使用 Cookie 的網站或服務,建議您在 Android 12 WebView 上測試流程。如果發現問題,您可能需要更新 Cookie,才能支援新的 SameSite 行為。

請留意登入和嵌入內容中的問題,以及登入流程、購買流程和其他驗證流程,當使用者從不安全的頁面開始,然後轉換至安全的頁面時,請留意這些流程中的問題。

如要使用 WebView 測試應用程式,您必須為要測試的應用程式啟用新的 SameSite 行為,方法是完成下列任一步驟:

  • 在測試裝置上手動啟用 SameSite 行為,方法是切換 WebView 開發人員工具中的 UI 標記 webview-enable-modern-cookie-same-site。

    您可以使用這種方法,在搭載 Android 5.0 (API 級別 21) 以上版本 (包括 Android 12) 的任何裝置上進行測試,並使用 WebView 89.0.4385.0 以上版本。

  • 請在 targetSdkVersion 前編譯應用程式,以針對 Android 12 (API 級別 31) 為目標。

    如果您採用這種做法,則必須使用搭載 Android 12 的裝置。

如要瞭解如何在 Android 上為 WebView 進行遠端偵錯,請參閱「開始使用 Android 裝置的遠端偵錯功能」。

其他資源

如要進一步瞭解 SameSite 現代行為,以及推出至 Chrome 和 WebView 的作業,請造訪 Chromium SameSite 更新頁面。如果您在 WebView 或 Chromium 中發現錯誤,可以透過公開的 Chromium 問題追蹤工具回報。

動作感應器有速率限制

為保護使用者的潛在私密資訊,如果您的應用程式指定 Android 12 以上版本,系統會限制特定動作感應器和位置感應器資料的重新整理率。

進一步瞭解感應器頻率限制

應用程式休眠

Android 12 擴充了 Android 11 (API 級別 30) 中導入的權限自動重設行為。如果應用程式指定的是 Android 12,而使用者已連續數月未與應用程式互動,系統會自動重設所有已授予的權限,並將應用程式設為「休眠」狀態。

詳情請參閱應用程式休眠模式指南。

資料存取稽核中的歸因聲明

Android 11 (API 級別 30) 推出的資料存取稽核 API 可讓您根據應用程式的用途建立歸因標記。這些代碼可讓您更輕鬆地判斷應用程式的哪個部分執行特定類型的資料存取作業。

如果應用程式指定 Android 12 以上版本,則必須在應用程式的資訊清單檔案中宣告這些歸因標記

ADB 備份限制

為保護私人應用程式資料,Android 12 會變更 adb backup 指令的預設行為。針對指定 Android 12 (API 級別 31) 以上版本為目標版本的應用程式,當使用者執行 adb backup 指令時,系統會從從裝置匯出的任何其他系統資料中排除應用程式資料。

如果您的測試或開發工作流程仰賴使用 adb backup 的應用程式資料,現在可以選擇在應用程式資訊清單檔案中將 android:debuggable 設為 true,匯出應用程式資料。

更安全的元件匯出功能

如果您的應用程式指定 Android 12 以上版本,且包含使用意圖篩選器活動服務廣播接收器,您必須為這些應用程式元件明確宣告 android:exported 屬性。

如果應用程式元件包含 LAUNCHER 類別,請將 android:exported 設為 true。在大多數情況下,請將 android:exported 設為 false

以下程式碼片段顯示服務的範例,其中包含意圖篩選器,且 android:exported 屬性設為 false

<service android:name="com.example.app.backgroundService"
         android:exported="false">
    <intent-filter>
        <action android:name="com.example.app.START_BACKGROUND" />
    </intent-filter>
</service>

Android Studio 中的訊息

如果應用程式包含使用意圖篩選器的活動、服務或廣播接收器,但未宣告 android:exported,則會顯示下列警告訊息,具體取決於您使用的 Android Studio 版本:

Android Studio 2020.3.1 Canary 11 以上版本

系統會顯示下列訊息:

  1. 資訊清單檔案中會顯示下列 Lint 警告:

    When using intent filters, please specify android:exported as well
    
  2. 嘗試編譯應用程式時,系統會顯示下列建構錯誤訊息:

    Manifest merger failed : Apps targeting Android 12 and higher are required \
    to specify an explicit value for android:exported when the corresponding \
    component has an intent filter defined.
    
舊版 Android Studio

如果您嘗試安裝應用程式,Logcat 會顯示以下錯誤訊息:

Installation did not succeed.
The application could not be installed: INSTALL_FAILED_VERIFICATION_FAILURE
List of apks:
[0] '.../build/outputs/apk/debug/app-debug.apk'
Installation failed due to: 'null'

待處理意圖的可變動性

如果應用程式以 Android 12 為目標版本,您必須為應用程式所建立的每個 PendingIntent 物件指定可變動性。這項額外要求可以提升應用程式的安全性。

測試待處理的意圖可變動性變更

如要判斷應用程式是否缺少可變動性宣告,請在 Android Studio 中尋找以下 Lint 警告

Warning: Missing PendingIntent mutability flag [UnspecifiedImmutableFlag]

不安全的意圖啟動

為提升平台安全性,Android 12 以上版本提供偵錯功能,可偵測意圖的不安全啟動行為。當系統偵測到這類不安全的啟動行為,就會發生 StrictMode 違規問題。

成效

前景服務啟動限制

指定 Android 12 以上版本為目標版本的應用程式,在背景執行時無法啟動前景服務 (少數特殊情況除外)。如果應用程式嘗試在背景執行時啟動前景服務,就會發生例外情況 (少數特殊情況除外)。

在應用程式於背景執行的情況下,建議您使用 WorkManager 安排及啟動加急作業。如要完成使用者要求且須及時處理的動作,請在確切的鬧鐘時間內啟動前景服務。

精確鬧鐘權限

為鼓勵應用程式節省系統資源,指定 Android 12 以上版本為目標版本,並設定精確鬧鐘的應用程式,必須具備「鬧鐘與提醒」功能,這項功能會顯示在系統設定的「特殊應用程式存取權」頁面中。

如要取得這項特殊的應用程式存取權,請在資訊清單中要求 SCHEDULE_EXACT_ALARM 權限。

精確鬧鐘應只用於面向使用者的功能。進一步瞭解設定精確鬧鐘的適用用途

停用行為變更

在準備將應用程式指定為 Android 12 的目標版本時,您可以暫時停用偵錯 版本變化版本中的行為變更,以利測試。如要這樣做,請完成下列任一步驟:

  • 在「開發人員選項」設定畫面中,選取「應用程式相容性變更」。在顯示的畫面中,輕觸應用程式名稱,然後關閉「REQUIRE_EXACT_ALARM_PERMISSION
  • 在開發機器的終端機視窗中執行下列指令:

    adb shell am compat disable REQUIRE_EXACT_ALARM_PERMISSION PACKAGE_NAME
    

通知彈跳床限制

當使用者與通知互動時,部分應用程式會啟動應用程式元件,最終啟動使用者最終看到並互動的活動,以回應輕觸通知的動作。這個應用程式元件又稱為通知彈跳床

為改善應用程式效能和使用者體驗,目標 Android 12 以上版本的應用程式無法透過服務或用來做為通知彈跳床的廣播接收器啟動活動。換句話說,使用者輕觸通知或通知中的動作按鈕後,您的應用程式就無法在服務或廣播接收器中呼叫 startActivity()

當應用程式嘗試透過做為通知彈跳床的服務或廣播接收器啟動活動時,系統會防止活動開始,且 Logcat 中顯示以下訊息:

Indirect notification activity start (trampoline) from PACKAGE_NAME, \
this should be avoided for performance reasons.

找出哪些應用程式元件會做為通知彈跳床

測試應用程式時,輕觸通知後,您可以找出哪些服務或廣播接收器在應用程式中充當通知彈跳點。如要找出這些服務或廣播接收器,請查看下列終端機指令的輸出內容:

adb shell dumpsys activity service \
  com.android.systemui/.dump.SystemUIAuxiliaryDumpService

輸出內容的某個部分包含「NotifInteractionLog」文字。本節包含用於識別因輕觸通知而啟動的元件所需的資訊。

請更新應用程式

如果您的應用程式是透過做為通知彈跳床的服務或廣播接收器啟動活動,請完成下列遷移步驟:

  1. 建立 PendingIntent 物件,並與使用者輕觸通知後看到的活動建立關聯。
  2. 使用您在上一個步驟中建立的 PendingIntent 物件,建立通知

如要識別活動來源,以便執行記錄等作業,請在發布通知時使用額外項目。如要使用集中式記錄功能,請使用 ActivityLifecycleCallbacksJetpack 生命週期觀察器

切換行為

測試可進行偵錯的應用程式版本時,您可以使用 NOTIFICATION_TRAMPOLINE_BLOCK 應用程式相容性標記啟用或停用這項限制。

備份與還原

在 Android 12 (API 級別 31) 上執行並指定此版本的應用程式,其備份與還原功能的運作方式會有所變更。Android 備份和還原有兩種形式:

  • 雲端備份:使用者資料會儲存在使用者的 Google 雲端硬碟中,以便日後在該裝置或新裝置上還原。
  • 裝置對裝置 (D2D) 轉移:使用者資料會從舊裝置直接傳送至新裝置,例如透過傳輸線。

如要進一步瞭解如何備份及還原資料,請參閱「使用自動備份功能備份使用者資料」和「使用 Android 備份服務備份鍵/值組合」。

D2D 傳輸功能變更

針對 Android 12 以上版本執行及指定 Android 12 以上版本的應用程式:

  • 使用 XML 設定機制指定包含和排除規則不會影響 D2D 傳輸,但仍會影響雲端備份和還原功能 (例如 Google 雲端硬碟備份)。如要指定 D2D 傳輸規則,您必須使用下一節說明的新設定。

  • 在部分裝置製造商的裝置上,指定 android:allowBackup="false" 會停用 Google 雲端硬碟備份功能,但不會停用應用程式的 D2D 傳輸功能。

新的包含和排除格式

在 Android 12 以上版本中運作且指定 Android 12 或以上版本的應用程式會針對 XML 設定使用不同的格式。這個格式會要求您分別為雲端備份和 D2D 轉移作業指定包含和排除規則,因此可明確區分 Google 雲端硬碟備份和 D2D 轉移作業的差異。

您也可以視需要使用此方法指定備份規則,在這種情況下,裝置會忽略先前使用的設定,且必須執行 Android 12 以上版本。搭載 Android 11 以下版本的裝置仍需要舊版設定。

XML 格式變更

以下是 Android 11 以下版本的備份與還原設定所使用的格式:

<full-backup-content>
    <include domain=["file" | "database" | "sharedpref" | "external" |
                     "root"] path="string"
    requireFlags=["clientSideEncryption" | "deviceToDeviceTransfer"] />
    <exclude domain=["file" | "database" | "sharedpref" | "external" |
                     "root"] path="string" />
</full-backup-content>

以下以粗體字顯示格式變更。

<data-extraction-rules>
  <cloud-backup [disableIfNoEncryptionCapabilities="true|false"]>
    ...
    <include domain=["file" | "database" | "sharedpref" | "external" |
                        "root"] path="string"/>
    ...
    <exclude domain=["file" | "database" | "sharedpref" | "external" |
                        "root"] path="string"/>
    ...
  </cloud-backup>
  <device-transfer>
    ...
    <include domain=["file" | "database" | "sharedpref" | "external" |
                        "root"] path="string"/>
    ...
    <exclude domain=["file" | "database" | "sharedpref" | "external" |
                        "root"] path="string"/>
    ...
  </device-transfer>
</data-extraction-rules>

如需更多資訊,請參閱使用自動備份功能備份使用者資料的指南中相應章節

應用程式的資訊清單標記

在資訊清單檔案中使用 android:dataExtractionRules 屬性,將應用程式指向新的 XML 設定。指向新的 XML 設定時,在搭載 Android 12 以上版本的裝置上,系統會忽略指向舊設定的 android:fullBackupContent 屬性。以下程式碼範例顯示新的資訊清單檔案項目:

<application
    ...
    <!-- The below attribute is ignored. -->
    android:fullBackupContent="old_config.xml"
    <!-- You can point to your new configuration using the new
         dataExtractionRules attribute . -->
    android:dataExtractionRules="new_config.xml"
    ...>
</application>

連線能力

藍牙權限

Android 12 導入了 BLUETOOTH_SCANBLUETOOTH_ADVERTISEBLUETOOTH_CONNECT 權限。有了這些權限,以 Android 12 為目標版本的應用程式就能更輕鬆地與藍牙裝置互動,尤其是不需要存取裝置位置資訊的應用程式。

如要讓裝置準備好指定 Android 12 以上版本,請更新應用程式的邏輯。請改為宣告舊版藍牙權限組合,宣告較新式的藍牙權限組合

同時進行點對點 + 網際網路連線

如果應用程式指定 Android 12 (API 級別 31) 以上版本,則支援同時進行點對點和網路連線的裝置,可同時與對等裝置和主要網路提供者網路建立 Wi-Fi 連線,讓使用者體驗更流暢。指定 Android 11 (API 級別 30) 以下版本的應用程式仍會發生舊版行為,也就是在連線至對等裝置之前,會先中斷主要 Wi-Fi 網路。

相容性

WifiManager.getConnectionInfo() 只能針對單一網路傳回 WifiInfo。因此,在 Android 12 以上版本中,API 的行為已發生以下變更:

  • 如果只有一個 Wi-Fi 網路可用,系統會傳回該網路的 WifiInfo
  • 如果有可用的 Wi-Fi 網路多於一個,且呼叫應用程式觸發對等端對連線,系統會傳回對應至對等端裝置的 WifiInfo
  • 如果有可用的 Wi-Fi 網路不只一個,且呼叫應用程式未觸發對等端連線,系統會傳回主要網際網路提供連線的 WifiInfo

為了在支援雙重 Wi-Fi 網路的裝置上提供更好的使用者體驗,我們建議所有應用程式 (尤其是會觸發點對點連線的應用程式) 都不要再呼叫 WifiManager.getConnectionInfo(),而是改用 NetworkCallback.onCapabilitiesChanged() 來取得與用於註冊 NetworkCallbackNetworkRequest 相符的所有 WifiInfo 物件。getConnectionInfo() 已於 Android 12 淘汰。

以下程式碼範例說明如何在 NetworkCallback 中取得 WifiInfo

Kotlin

val networkCallback = object : ConnectivityManager.NetworkCallback() {
  ...
  override fun onCapabilitiesChanged(
           network : Network,
           networkCapabilities : NetworkCapabilities) {
    val transportInfo = networkCapabilities.getTransportInfo()
    if (transportInfo !is WifiInfo) return
    val wifiInfo : WifiInfo = transportInfo
    ...
  }
}

Java

final NetworkCallback networkCallback = new NetworkCallback() {
  ...
  @Override
  public void onCapabilitiesChanged(
         Network network,
         NetworkCapabilities networkCapabilities) {
    final TransportInfo transportInfo = networkCapabilities.getTransportInfo();
    if (!(transportInfo instanceof WifiInfo)) return;
    final WifiInfo wifiInfo = (WifiInfo) transportInfo;
    ...
  }
  ...
};

mDNSResponder 原生 API

Android 12 會變更應用程式使用 mDNSResponder 原生 API 與 mDNSResponder 守護程序互動的時間。之前,當應用程式在網路上註冊服務並呼叫 getSystemService() 方法時,系統的 NSD 服務會啟動 mDNSResponseer Daemon,即使應用程式尚未呼叫任何 NsdManager 方法也一樣。接著,Daemon 會將裝置訂閱到所有節點的多點傳播群組,導致系統更頻繁地喚醒並使用額外電源。為盡量減少電池用量,在 Android 12 以上版本中,系統現在只會在需要 NSD 事件時啟動 mDNSResponder 守護程序,並在之後停止。

由於這項變更會影響 mDNSResponder 守護程式可用的時間,因此假設在呼叫 getSystemService() 方法後,mDNSResponder 守護程式會啟動的應用程式,可能會收到系統傳送的訊息,指出 mDNSResponder 守護程式無法使用。使用 NsdManager 且未使用 mDNSResponder 原生 API 的應用程式不會受到這項異動影響。

供應商程式庫

供應商提供的原生共用程式庫

根據預設,如果應用程式是以 Android 12 (API 級別 31) 以上版本為目標版本,則無法存取由晶片供應商或裝置製造商提供的非 NDK 原生共用程式庫。只有在使用 <uses-native-library> 標記明確要求時,才能存取程式庫。

如果應用程式指定 Android 11 (API 級別 30) 以下版本,則不需要 <uses-native-library> 標記。在這種情況下,任何原生共享程式庫均可存取,無論其是否為 NDK 程式庫。

更新非 SDK 限制

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

如果您的應用程式並未以 Android 12 為目標版本,則此處所述的某些變更可能不會立即對您造成影響。不過,雖然您目前可使用某些非 SDK 介面 (視應用程式的目標 API 級別而定),但使用任何非 SDK 方法或欄位時,一律可能會導致應用程式停止運作。

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

如要進一步瞭解此 Android 版本中的變更,請參閱 Android 12 的非 SDK 介面限制更新。如要進一步瞭解非 SDK 介面的一般資訊,請參閱非 SDK 介面的限制