Android 10 - 文件提供者 - 权限拒绝:读取android.support.v4.content.FileProvider URI

8
我试图生成一个 CSV 文件(加速度计数据),保存到我的 Context.getFilesDir() 中,然后使用 FileProvider 共享到 Google Drive。但问题是当我尝试分享时,会不断出现错误 Writing exception to parceljava.lang.SecurityException: Permission Denial: reading android.support.v4.content.FileProvider uri content://...csv requires the provider be exported, or grantUriPermission()
我已经尝试了以下方法:
  1. 添加了 grantUriPermission()Intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION),像其他人建议的那样。
  2. 我尝试将文件写入 Environment.getExternalStorageDirectory(),但由于我的手机没有外部存储,所以这个方法无法实现。
  3. 我不能导出提供者,因为这也会引发错误。
那么我该怎么办呢?或者有更好的方法可以生成和共享 CSV 文件吗?感谢您们的建议。以下是我的代码片段:
在 AndroidManifest.xml 中:
    <application...>
        ...

        <provider
            android:name="android.support.v4.content.FileProvider"
            android:grantUriPermissions="true"
            android:authorities="com.example.FileProvider"
            android:exported="false">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/provider_paths"/>
        </provider>

    </application>

在 provider_path.xml 文件中。
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <files-path name="files" path="."/>
</paths>

在mainActivity.java文件中

    private void onShareCSV() {

        Uri path = FileProvider.getUriForFile(this, "com.example.FileProvider", new File(filePath + fileName));

        Log.d(TAG, "onShareCSV: " + path);
        Intent shareIntent = new Intent("com.example.FileProvider").setData(path);

        List<ResolveInfo> resInfoList = getPackageManager().queryIntentActivities(shareIntent, PackageManager.MATCH_DEFAULT_ONLY);
        for (ResolveInfo resolveInfo : resInfoList) {
            String packageName = resolveInfo.activityInfo.packageName;
            getApplicationContext().grantUriPermission(packageName, path, Intent.FLAG_GRANT_READ_URI_PERMISSION);
        }

        shareIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        shareIntent.setAction(Intent.ACTION_SEND);
        shareIntent.putExtra(Intent.EXTRA_TEXT, "This is a CSV I'm sharing.");
        shareIntent.putExtra(Intent.EXTRA_STREAM, path);
        shareIntent.setType("text/csv");
        startActivity(Intent.createChooser(shareIntent, "Share..."));
        this.setResult(RESULT_OK, shareIntent);
    }

以下是错误信息

2020-02-20 23:45:02.751 21484-21503/com.example.charleschung.mci_pa1 E/DatabaseUtils: Writing exception to parcel
    java.lang.SecurityException: Permission Denial: reading android.support.v4.content.FileProvider uri content://com.example.FileProvider/files/sensor-1582260302226.csv from pid=21581, uid=1000 requires the provider be exported, or grantUriPermission()
        at android.content.ContentProvider.enforceReadPermissionInner(ContentProvider.java:729)
        at android.content.ContentProvider$Transport.enforceReadPermission(ContentProvider.java:602)
        at android.content.ContentProvider$Transport.query(ContentProvider.java:231)
        at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:104)
        at android.os.Binder.execTransactInternal(Binder.java:1021)
        at android.os.Binder.execTransact(Binder.java:994)

Uri path = FileProvider.getUriForFile(this........。请告诉我path.toString()的值。 - blackapps
@blackapps 感谢您的回复,我已重新格式化代码片段。路径值为 'content://com.example.FileProvider/files/sensor-1582271002304.csv'。 - Charles Chung
请删除以下代码:List<ResolveInfo> resInfoList = getPackageManager().queryIntentActivities(shareIntent, PackageManager.MATCH_DEFAULT_ONLY); for (ResolveInfo resolveInfo : resInfoList) { String packageName = resolveInfo.activityInfo.packageName; getApplicationContext().grantUriPermission(packageName, path, Intent.FLAG_GRANT_READ_URI_PERMISSION); } 您不需要它。 - blackapps
谢谢,但是删除了那段代码后仍然没有改变。我还能做些什么吗? - Charles Chung
请返回答案:stackoverflow.com/a/74052302/987762 - kalandar
3个回答

1

我终于解决了。原来我的URI构造有些问题。我把它改成了

Context context = getApplicationContext();
File filelocation = new File(getFilesDir(), fileName);
Uri path = FileProvider.getUriForFile(context, "com.example.charleschung.mci_pa1", filelocation);

它可以工作。希望能有所帮助 :)


仍然对我不起作用。 - FabioR

0

首先,您应该检查构建版本代码

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                        Uri path = FileProvider.getUriForFile(this,getApplicationContext().getPackageName() +  ".FileProvider", new File(filePath + fileName)); 

                    } else{
                        Uri path =Uri.fromFile(new File(filePath + fileName)) 
                    }

希望它有所帮助


-4
我也遇到了这个问题,但我的情况与你的略有不同,我试图将文件保存在一个文件夹中。我缺少的是 Android 10 的权限,所以我在清单中添加了以下行:

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

还需要在android:requestLegacyExternalStorage="true"中添加一行

同时添加运行时权限

Manifest.permission.ACCESS_MEDIA_LOCATION


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接