范围存储:getExternalStoragePublicDirectory 和 setDestinationInExternalPublicDir

问题描述

对于 Scoped Storage,getExternalStoragePublicDirectory 已被弃用,Android 文档说

方法已在 API 级别 29 中弃用。为了提高用户隐私,不推荐直接访问共享/外部存储设备。当应用以 Build.VERSION_CODES.Q 为目标时,应用无法再直接访问从此方法返回的路径。通过迁移到 Context#getExternalFilesDir(String)、MediaStore 或 Intent#ACTION_OPEN_DOCUMENT 等替代方案,应用可以继续访问存储在共享/外部存储上的内容

但是 setDestinationInExternalPublicDir 没有被弃用。目前我正在使用downloadManager下载pdf并将其保存在以下载目录为根目录的文件夹中(通过将目录作为DIRECTORY_DOWNLOADS到setDestinationInExternalPublicDir)。然后我使用已弃用的 FILE 方法创建一个 getExternalStoragePublicDirectory() 对象,并为所有适当的应用程序触发一个意图来处理 PDF 查看。

我的问题是,当我将 requestLegacyExternalStorage 标志保持为 true 时,它​​工作正常(我的目标 sdk 是 29)。但是,当我省略它时,它有时可以正常工作,但有时 PDF 查看器会打开文件然后立即关闭它。我试图在我的应用程序本身中读取文件以查看出了什么问题,但出现了拒绝访问异常。这是有道理的,因为对于范围存储,getExternalStoragePublicDirectory() 返回不可访问的路径。但是,这种间歇性如何像其他时候一样正常工作?

补充问题,我不明白为什么我的设置 requestLegacyExternalStorage 标志会产生任何影响。我所做的只是使用 setDestinationInExternalPublicDir 中不推荐使用的 DownloadManager 方法将 pdf 保存到 Download 目录中的子文件夹中。然后我创建一个文件(是的,我通过一个已弃用的 getExternalStoragePublicDirectory 知道它可能无法访问 - 但我不在乎,因为我没有访问它),从中获取 uri 并激发意图 - 因此行为不应该仅仅是取决于 pdf 查看器应用程序是否可以访问旧文件夹(也就是说,如果它们在 APIrequestLegacyExternalStorage 设置为 true)? - 这只是为了理解这个概念,当然我需要处理这个并将其存储在某个可访问的位置。

这带来了我的最后一个查询。为了处理这个问题,我可以选择使用 MediaStore.Downloads 来下载下载目录中的文件。我还可以使用 DownloadManager 在该位置下载它吗? Android 文档说,即使 scoped storage 触发 Intent 并授予其他应用程序权限,我们只需要授予此权限:FLAG_GRANT_READ_URI_PERMISSION (我已经在做)。那么这是否意味着意图接收应用程序可以通过这种方式访问​​所有路径?如果没有,那么如果 MediaStore.Downloads 存储在 Downloads 目录并且其他应用程序可以通过 SAF 访问它,那么它不应该通过意图触发访问吗?如果是,为什么我当前的实施间歇性失败。

任何线索都将受到高度赞赏。谢谢。

解决方法

然后我使用已弃用的 getExternalStoragePublicDirectory() 方法创建一个 FILE 对象,并为所有适当的应用程序触发一个意图来处理 PDF 查看。

那不是要走的路。

相反,您应该为 ACTION_DOWNLOAD_COMPLETE 注册一个广播接收器。

在 onReceive() 中,您可以获得下载文件的 uri。

将该 uri 与您的意图 ACTION_VIEW 的授予读取权限标志一起使用。