Android APIv29 FileNotFoundException EACCES(权限被拒绝)

Android APIv29 FileNotFoundException EACCES (Permission denied)

我在为 targetSdkVersion v29 构建时无法访问存储。

这是我的 gradle 配置:

    compileSdkVersion 29
    buildToolsVersion "29.0.2"
    ...
        minSdkVersion 15
        targetSdkVersion 29

请注意,已授予 WRITE_EXTERNAL_STORAGE 权限,并且在为 targetSdkVersion 28 构建时相同的设置工作正常。

这是我的实现:

        val outputFolder = File(baseFolder + File.separator + "Output Folder")
        if (!outputFolder.exists()) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                Files.createDirectory(outputFolder.toPath()) //This allways returns false with targetSdkVersion 29
            } else {
                if (!outputFolder.mkdirs()) {
                    Log.e("SaveRaw", "Unable to create folder for audio recording")
                }
            }
        }

        outputFile = File("$baseFolder/Output Folder/$filename")
        try {
            fileOutputStream = FileOutputStream(outputFile)
        } catch (e: FileNotFoundException) {
            e.printStackTrace() // allways throwing exception here, even if Output Folder exists 
        }

这里是个例外:

W/System.err: java.io.FileNotFoundException: /storage/emulated/0/Chirp Auto Tester/2019_10_17 10:44:43.raw: open failed: EACCES (Permission denied)
W/System.err:     at libcore.io.IoBridge.open(IoBridge.java:496)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:235)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:186)

希望有人能给出答案,我在这里缺少什么?

更新:

这里是 baseFolder 的来源。请注意,getExternalStorageDirectory 是一种已弃用的方法。

        val baseFolder: String = if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
            Environment.getExternalStorageDirectory().absolutePath
        } else {
            context.filesDir.absolutePath
        }

谢谢

Starting with Android 11 the storage permission is getting revoked and developers would need to consider alternative ways of accessing the storage they need either through SAF or Media Store。目前,您可以通过在应用程序标签内的清单中添加以下内容来继续使用您正在使用的内容:

android:requestLegacyExternalStorage="true"

您可能需要考虑将 minSDK 更改为 19,并使用 getExternalFilesDir() 获取不需要任何权限的路径。

当您的应用面向 Build.VERSION_CODES.Q 时,应用不再可以直接访问从此方法返回的路径。

    val path = getExternalFilesDir(Environment.DIRECTORY_PICTURES.toString() + 
    File.separator + "Output Folder")
    val file = File(getExternalFilesDir(null), "filename.jpg")

有关详细信息,请访问 Docs

随着 Android 版本 10(API 级别 29)引入了分区存储的概念。根据 Docs-“为了让用户更好地控制他们的文件并限制文件混乱,针对 Android 10(API 级别 29)及更高级别的应用程序被授予对外部存储或范围存储的范围访问权限, 默认情况下。此类应用只能访问外部存储上的 app-specific 目录,以及该应用创建的特定类型的媒体。"

对于 Android 版本 10,您可以选择 opt-out 范围存储,方法是在应用程序标签 AndroidManifest.xml 中添加 android:requestLegacyExternalStorage="true"

这是一个临时解决方案,仅适用于 android 版本 10,不适用于更高版本。还要确保您请求 run-time 许可 read/write 外部存储。

Android 11 将简单地忽略 android:requestLegacyExternalStorage="true",因为它是 Android 的临时解决方案< 11 不破坏旧应用程序。

 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />

以及清单文件的应用程序标签中:

 android:requestLegacyExternalStorage="true"

也像这样在清单文件的应用程序标签中声明提供者:

 <provider
        android:name="androidx.core.content.FileProvider"
        android:authorities="${applicationId}.provider"
        android:exported="false"
        android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/provider_path" />
    </provider>

在 res 文件夹中创建一个名为 xml 的文件夹。

现在在您刚刚创建的 xml 文件夹中创建一个名为 provider_path.xml 的 xml 文件并将以下代码粘贴到 xml 文件中:

<?xml version="1.0" encoding="utf-8"?>

<path>
<external-path
    name="external_files"
    path="." />

现在在您的 activity 中:

File file = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS)+"/"+"sample.pdf");
                    if(file.exists()){
                        Uri uri = FileProvider.getUriForFile(context, "com.example.www"+".provider",file);
                        Intent i = new Intent(Intent.ACTION_VIEW);
                        i.setDataAndType(uri, "application/pdf");
                        i.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP|Intent.FLAG_GRANT_READ_URI_PERMISSION);
                        context.startActivity(i);
                    }

将 com.example.www 替换为您的应用程序包名称。