Android Q (10) 请求获得访问所有存储的权限。范围存储

use*_*041 23 android android-10.0 scoped-storage

在 Android 10 中,需要访问存储的应用程序需要请求访问具体路径的权限。但是像文件浏览器这样的应用程序可以请求访问根存储的权限并获得对所有存储的读/写权限。这就是我正在努力做的。

根据Android,我们必须ACTION_OPEN_DOCUMENT_TREE为此使用。我遇到的问题是一切似乎都正确,但未向应用授予权限。

  private void askAndroid10Perm()
    {
        Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE);
        intent.addFlags(
                Intent.FLAG_GRANT_READ_URI_PERMISSION
                        | Intent.FLAG_GRANT_WRITE_URI_PERMISSION
                        | Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION
                        | Intent.FLAG_GRANT_PREFIX_URI_PERMISSION);
        startActivityForResult(intent, REQUEST_CODE_OPEN_DOCUMENT_TREE);
    }
Run Code Online (Sandbox Code Playgroud)

在这里,用户可以看到 Android 文件树,并选择主存储,单击以授予权限。“它将允许 - 应用程序名称 - 可以完全访问当前存储在此位置下的所有文件,以及存储在此处的任何未来内容”-> 允许

然后:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    switch (requestCode) {
        case REQUEST_CODE_OPEN_DOCUMENT_TREE:
            if (resultCode == Activity.RESULT_OK) {
                Uri treeUri = data.getData();
                int takeFlags = data.getFlags();
                takeFlags &= (Intent.FLAG_GRANT_READ_URI_PERMISSION |
                        Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {

                    Log.i("TAG", "takePersistableUriPermission: " + treeUri);
                    this.getContentResolver().takePersistableUriPermission(treeUri, takeFlags);

                }

            }
    }
}
Run Code Online (Sandbox Code Playgroud)

日志:

takePersistableUriPermission: content://com.android.externalstorage.documents/tree/primary%3A

然后该应用程序仍然没有访问 root 的权限。

打开失败:EACCES(权限被拒绝)

当然,我有:

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
Run Code Online (Sandbox Code Playgroud)

由用户授予。

D_K*_*D_K 39

在清单中添加行

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:requestLegacyExternalStorage="true" //Add this Line
    android:label="@string/app_name">

---------<Activity, Sevices or Recivers>----------

</application>
Run Code Online (Sandbox Code Playgroud)

  • 这只是暂时的,请参阅 https://developer.android.com/training/data-storage/use-cases#opt-out-scoped-storage (3认同)

toh*_*rad 11

您必须在 AndroidManifest.xml 中添加这一行

android:requestLegacyExternalStorage="true"
Run Code Online (Sandbox Code Playgroud)


use*_*041 7

问题是我使用 File 对象来访问文件,但权限只有在我们使用 DocumentFile 对象来操作文件时才有效。

使用 DocumentFile 而不是 File 解决。

使用 DocumentFile 有多个负面影响,例如性能以及您无法访问 apache 公共库,但没有其他选择。

  • 例如,有一些用于 java 的库,您可以在“文件”对象中复制、剪切或压缩目录。DocumentFile 不存在此库。 (2认同)