概述
我们正在 Android 平台上进行多项变更来增强用户隐私和平台安全性,旨在为用户提供更安全的体验。以 Android 11 (API 级别 30) 或更高版本为目标的应用默认将只能获取过滤后的已安装应用列表。如需访问过滤后列表以外的应用,则需要在应用内的 Android manifest 中使用 <queries> 元素声明需要与之交互的应用。本文将介绍适应此特性的最佳实践。
过滤后的已安装应用列表
https://developer.android.google.cn/training/basics/intents/package-visibility#automatic
查询应用并与之交互
您可以通过以下几种方式查询应用并与之交互:
如果您知道想要查询或与之交互的特定应用集,请将其软件包名称包含在 <queries> 元素内的一组 <package> 元素中。
<manifest package="com.example.game">
<queries>
<package android:name="com.example.store" />
<package android:name="com.example.services" />
</queries>
...
</manifest>
如果您的应用需要查询或与一组具有特定用途的应用交互,但您可能不知道要添加的具体软件包名称,您可以将 intent 过滤器签名列在您的 <queries> 元素中。然后,您的应用便可发现具有匹配的 <intent-filter> 元素的应用。
<manifest package="com.example.game">
<queries>
<intent>
<action android:name="android.intent.action.SEND" />
<data android:mimeType="image/jpeg" />
</intent>
</queries>
...
</manifest>
如果您需要查询 Content Provider,但不知道具体的软件包名称,则可以在 <provider> 元素中声明该提供程序授权。
<manifest package="com.example.suite.enterprise">
<queries>
<provider android:authorities="com.example.settings.files" />
</queries>
...
</manifest>
软件包
https://developer.android.google.cn/training/basics/intents/package-visibility#package-name
intent 过滤器签名
https://developer.android.google.cn/training/basics/intents/filters
Content Provider
https://developer.android.google.cn/guide/topics/providers/content-provider-basics#ContentURIs
我们建议通过仅查询您需要与之交互的软件包来尽可能减少数据。QUERY_ALL_PACKAGES 或同等广泛的 <intent> 元素应当仅由需要此级别信息的应用使用。我们新增的软件包可见性政策为新推出的 QUERY_ALL_PACKAGES 权限引入了一个审批流程,用于控制对设备上已安装应用清单的访问。您可以观看以下视频或阅读更多政策更新。
Google Play 政策更新|2021 年 3 月
腾讯视频链接
https://v.qq.com/x/page/h3238tkvuu6.html
Bilibili 视频链接
https://www.bilibili.com/video/BV1qX4y1g75U/
政策更新
https://support.google.com/googleplay/android-developer/answer/9934569?hl=en&ref_topic=9877065
Activity 标记
大多数常见用例都不需要您的应用具有广泛的软件包可见性。对于许多场景,您可以使用 startActivity(),并在没有应用可以打开此 intent 时捕获异常。
try {
val intent = Intent(ACTION_VIEW, Uri.parse(url)).apply {
addCategory(CATEGORY_BROWSABLE)
}
startActivity(intent)
} catch (e:ActivityNotFoundException) {
Snackbar.make(it,"Activity Not Found",Snackbar.LENGTH_LONG).show()
}
尽管您可以启动没有目标可见性的任何 Activity,但由于它是一个隐式 intent,您在启动之前无法查询它的可用性,也无法了解将启动哪个特定的应用。如果您在它不解析的情况下启动,将收到通知。为了解决这一问题,您可以使用 intent 标记。
使用标记的常见示例是自定义标签页,自定义标签页让应用可以自定义浏览器的外观。链接将在非浏览器应用 (如果有) 中正确打开,而标记则可以在开发者希望能够自由选择 "自定义标签页" 浏览器的高级用例中提供帮助。
隐式 intent
https://developer.android.google.cn/guide/components/intents-filters#Types
自定义标签页
https://developer.chrome.com/docs/android/custom-tabs/overview/
FLAG_ACTIVITY_REQUIRE_NON_BROWSER
只有 intent 解析为非浏览器结果时,此标记才会启动它。如果此类结果不存在,将抛出 ActivityNotFoundException,然后,您的应用可以在自定义标签页中打开该网址。
val intent = Intent(ACTION_VIEW, Uri.parse(url)).apply {
// The URL should either launch directly in a non-browser app (if it's
// the default), or in the disambiguation dialog.
addCategory(CATEGORY_BROWSABLE)
flags = FLAG_ACTIVITY_NEW_TASK or FLAG_ACTIVITY_REQUIRE_NON_BROWSER
}
FLAG_ACTIVITY_REQUIRE_NON_BROWSER
https://developer.android.google.cn/reference/android/content/Intent#FLAG_ACTIVITY_REQUIRE_NON_BROWSER
ActivityNotFoundException
https://developer.android.google.cn/reference/android/content/ActivityNotFoundException
如果一个 intent 包含此标记,则在调用直接启动浏览器应用或者向用户显示一个消歧对话框 (唯一选项是浏览器应用) 时,调用 startActivity() 会导致抛出 ActivityNotFoundException。要详细了解标记,请参阅基于用例配置软件包可见性:
https://developer.android.google.cn/training/basics/intents/package-visibility-use-cases#let-non-browser-apps-handle-urls
自定义共享表单
建议使用系统提供的共享表单代替自定义表单。无需应用可见性,您也可以自定义系统共享表单。请参阅文档了解更多信息:
https://developer.android.google.cn/reference/android/content/Intent#ACTION_CHOOSER
调试软件包可见性
您可以轻松检查 manifest,了解是否包括了所有 queries。为此,请访问 manifest 文件并选择 Merged Manifest。
您也可以启用软件包过滤的日志消息,了解默认可见性对您的应用有何影响:
$ adb shell pm log-visibility --enable YOUR_PACKAGE_NAME
后续步骤
有关软件包可见性的详细信息,您可以参阅以下资源:
文档: 软件包可见性
https://developer.android.google.cn/training/package-visibility
Android 11 中的软件包可见性
乐享编码!
推荐阅读
点击屏末 | 阅读原文 | 即刻查看有关软件包可见性的更多信息
最后
以上就是魁梧石头为你收集整理的政策更新 | 开发者如何处理软件包可见性的全部内容,希望文章能够帮你解决政策更新 | 开发者如何处理软件包可见性所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复