我试图用相机拍照,但我得到以下错误:

FATAL EXCEPTION: main
Process: com.example.marek.myapplication, PID: 6747
java.lang.IllegalArgumentException: Failed to find configured root that contains /storage/emulated/0/Android/data/com.example.marek.myapplication/files/Pictures/JPEG_20170228_175633_470124220.jpg
    at android.support.v4.content.FileProvider$SimplePathStrategy.getUriForFile(FileProvider.java:711)
    at android.support.v4.content.FileProvider.getUriForFile(FileProvider.java:400)
    at com.example.marek.myapplication.MainActivity.dispatchTakePictureIntent(MainActivity.java:56)
    at com.example.marek.myapplication.MainActivity.access$100(MainActivity.java:22)
    at com.example.marek.myapplication.MainActivity$1.onClick(MainActivity.java:35)

AndroidManifest.xml:

<provider
        android:name="android.support.v4.content.FileProvider"
        android:authorities="com.example.marek.myapplication.fileprovider"
        android:enabled="true"
        android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/file_paths" />
</provider>

Java:

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    // Ensure that there's a camera activity to handle the intent
    if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
        // Create the File where the photo should go
        File photoFile = null;
        try {
            photoFile = createImageFile();
        } catch (IOException ex) {
            Toast.makeText(getApplicationContext(), "Error while saving picture.", Toast.LENGTH_LONG).show();
        }
        // Continue only if the File was successfully created
        if (photoFile != null) {
            Uri photoURI = FileProvider.getUriForFile(this,
                    "com.example.marek.myapplication.fileprovider",
                    photoFile);
            takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);
            startActivityForResult(takePictureIntent, REQUEST_TAKE_PHOTO);
        }
    }

file_paths.xml

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <files-path name="my_images" path="images/"/>
</paths>

我一整天都在搜索这个错误,试图理解FileProvider,但我不知道这个错误消息试图告诉我什么。如果你想要更多信息/代码,请在评论中告诉我。


当前回答

我的问题是,我在不同类型的文件路径中有重叠的名称,就像这样:

<cache-path
    name="cached_files"
    path="." />
<external-cache-path
    name="cached_files"
    path="." />

在我将名称(“cached_files”)更改为唯一后,我摆脱了错误。我的猜测是,这些路径存储在一些HashMap或不允许重复的东西中。

其他回答

我发现至少您没有在file_paths.xml中提供与其他文件相同的路径。 所以请确保在3个地方提供完全相同的包名或路径,包括:

Android:权限属性在manifest中 file_paths.xml中的Path属性 在调用FileProvider.getUriForFile()时参数。

我的问题是,我在不同类型的文件路径中有重叠的名称,就像这样:

<cache-path
    name="cached_files"
    path="." />
<external-cache-path
    name="cached_files"
    path="." />

在我将名称(“cached_files”)更改为唯一后,我摆脱了错误。我的猜测是,这些路径存储在一些HashMap或不允许重复的东西中。

我已经花了5个小时了。

我已经尝试了上面所有的方法,但这取决于你的应用程序当前使用的存储。

https://developer.android.com/reference/android/support/v4/content/FileProvider#GetUri

在尝试代码之前,请检查文档。

对我来说 因为文件路径子目录将是Context.getFilesDir()。 奇怪的是,Context.getFilesDir()注释了另一个子目录。

我要找的是

数据用户/ 0 / com . psh。mTest / app_imageDir / 20181202101432629 png。

Context.getFilesDir()

了 数据用户/ 0 / com . psh mTest / files。

标签应该是

....files-path name="app_imageDir" path="../ app_imageDir”……

那就有用了!!

我的问题是错误的文件名: 我在res/xml下创建file_paths.xml,而资源设置为Manifest中的provider_paths.xml:

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

我将provider_paths更改为file_paths,问题解决了。

检查您的设备提供了多少存储空间—不支持从辅助存储空间共享文件。查看FileProvider.java源代码(from support-core-utils 25.3.1):

            } else if (TAG_EXTERNAL_FILES.equals(tag)) {
                File[] externalFilesDirs = ContextCompat.getExternalFilesDirs(context, null);
                if (externalFilesDirs.length > 0) {
                    target = externalFilesDirs[0];
                }
            } else if (TAG_EXTERNAL_CACHE.equals(tag)) {
                File[] externalCacheDirs = ContextCompat.getExternalCacheDirs(context);
                if (externalCacheDirs.length > 0) {
                    target = externalCacheDirs[0];
                }
            }

所以,他们只取第一个存储空间。

此外,您还可以看到getExternalCacheDirs()用于通过ContextCompat接口获取存储列表。请参阅文档了解它的限制(例如,它被告知不能识别USB闪存)。最好的方法是自己对这个API的存储列表进行一些调试输出,这样就可以检查存储的路径与传递给getUriForFile()的路径是否匹配。

谷歌的问题跟踪器中已经分配了一张票(截至06-2017年),要求支持多个存储。最后,我也找到了一个关于这个问题的问题。