将文件写入 Android Q 中可公开访问的文档文件夹 - Scoped Storage

Jam*_*son 8 android mediastore android-10.0 scoped-storage

背景

迁移到 Android 后 QI 无法再找到合适的方式来获得对文档文件夹的写入权限 ( /storage/emulated/0/Documents)

在任何人提到有关范围存储的许多其他问题之前,我已经阅读了其中的许多问题,并且根据我目前所见,所有解决方案都使用特定于应用程序的目录或仅访问媒体目录(而不是文档文件夹访问权限)。

根据我在 Android QI 中的理解,可以选择:

  • 使用只能由我的应用程序(或我授予权限的应用程序)访问的特定于应用程序的目录
  • 允许用户选择他们想要存储文件的位置(我认为这可以是一个公共位置) ACTION_OPEN_DOCUMENT_TREE

问题

我开发的应用程序用于对主题进行测试,来自测试的数据会自动存储在可公开访问的 Documents 文件夹中,有点像: /storage/emulated/0/Documents/myAppName/subjectName/testData-todaysDate.pdf

当用户想要访问测试数据时,他们将智能手机插入计算机并导航到文档文件夹,其余的就很明显了。出于这个原因,我必须使用可公开访问的存储。还想象他们想通过另一个应用程序打开手机上的测试数据,同样的交易!

我正在寻找的解决方案

所以我正在寻找的解决方案需要能够做到以下几点:

  • 不多次请求许可(一次就可以)
  • 无需用户提示即可自动保存测试数据(因为将进行数百次测试)
  • 保存到公共文档文件夹,例如 /storage/emulated/0/Documents/
  • 不涉及用户选择目录所以不 ACTION_OPEN_DOCUMENT_TREE
  • 理想情况下,数据是持久的,因此卸载应用程序不会导致数据丢失

我知道 Android Q 中的这些变化是为了让用户更多地“了解应用程序如何访问他们的数据”,但是一旦他们了解您的应用程序如何使用您的数据,就不会有问题。

bla*_*pps 14

仅适用于 Android Q。

String collection = "content://media/1c11-2404/file"; // One can even write to micro SD card
String relative_path = "Documents/MyFolder";

Uri collectionUri = Uri.parse(collection);

ContentValues contentValues = new ContentValues();

contentValues.put(MediaStore.MediaColumns.DISPLAY_NAME, displayName);
contentValues.put(MediaStore.MediaColumns.MIME_TYPE, mimeType);
contentValues.put(MediaStore.MediaColumns.SIZE, filesize);
contentValues.put(MediaStore.MediaColumns.DATE_MODIFIED, modified );
contentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, relative_path);
contentValues.put(MediaStore.MediaColumns.IS_PENDING, 1)

Uri fileUri = context.getContentResolver().insert(collectionUri, contentValues);
Run Code Online (Sandbox Code Playgroud)

拥有一个 uri 可以打开一个输出流来写入文件的内容。

  • @blackapps:好的,我可以确认这有效。对于外部存储,您可以使用 `content://media/external/file` (也可以通过 `MediaStore.Files.getContentUri("external")` 获得)。对于可移动存储,您需要存储 ID(代码片段中的“1c11-2404”)。我怀疑这是否有效,所以如果这项技术在未来版本的 Android 中出现问题,我不会感到震惊。但是,这是一个很棒的发现——感谢分享! (2认同)
  • `OutputStream = getContentResolver().openOutputStream(fileUri);`。 (2认同)