最近做到一个用户头像裁剪上传的功能,因为项目的target是7.0的,所以遇到了一个应用件共享文件的问题,总是报
android.os.FileUriExposedException:
这个错误,然后查了一下资料,解决了问题。
首先 我们要理解问题的根源所在:
对于面向 Android N 的应用,Android 框架执行的 StrictMode,API 禁止向您的应用外公开 file://URI。如果一项包含文件 URI 的 Intent 离开您的应用,应用失败,并出现 FileUriExposedException异常。
若要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider类。 如需有关权限和共享文件的更多信息,请参阅共享文件。
FileProvider用在什么地方
A content URI allows you to grant read and write access using temporary access permissions. When you create an Intent containing a content URI, in order to send the content URI to a client app, you can also call Intent.setFlags() to add permissions. These permissions are available to the client app for as long as the stack for a receiving Activity is active. For an Intent going to a Service, the permissions are available as long as the Service is running.
使用临时授权的方式允许读写content URI。当你创建一个Intent包含content URI,为了发送content URI给其他app,使用Intent.setFlags()添加权限。其他app的activity栈只要是活动的,就能访问权限,service也一样。
现在我们应该知道问题原因了,接下来开始使用FIleProvider解决问题
以我的项目中的问题为例
一、先在Manifest中定义一个FileProvider
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.nbeebank.licaishi.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/imagepaths" />
</provider>
name可以使用系统定义的 android.support.v4.content.FileProvider,也可重写FileProvider methods。
- android:authorities 字段指定了希望使用的Authority,该Authority针对于FileProvider所生成的content URI。
authorities可以使用 包名 +"fileprovider"的方式命名。
<provider>下的<meta-data>指向了一个XML文件,该文件指定了我们希望共享的目录路径。“android:resource”属性字段是这个文件的路径和名字(无“.xml”后缀)
接下定义一下上边 resourse指定的共享文件的路径
在res/xml/imagepaths.xml目录下创建新文件(名字要和上边的对应,因为上边引用的路径就是这个),内容如下:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path
name="images_external"
path="." />
该方式提供在外部存储区域根目录下的文件。它对应Environment.getExternalStorageDirectory返回的路径:
eg:”/storage/emulated/0”;
<files-path
name="images_files"
path="." />
该方式提供在应用的内部存储区的文件/子目录的文件。它对应Context.getFilesDir返回的路径:
eg:”/data/data/包名/files”。
<cache-path
name="images_cache"
path="." />
该方式提供在应用的内部存储区的缓存子目录的文件。它对应getCacheDir返回的路径:
eg:“/data/data/包名/cache”;
<external-files-path
name="images_external_files"
path="." />
该方式提供在应用的外部存储区根目录的下的文件。它对应Context#getExternalFilesDir(String) Context.getExternalFilesDir(null)返回的路径。
eg:”/storage/emulated/0/Android/data/包名/files”
<external-cache-path
name="images_external_cache"
path="." />
该方式提供在应用的外部缓存区根目录的文件。它对应Context.getExternalCacheDir()返回的路径。
eg:”/storage/emulated/0/Android/data/包名/cache”。
<root-path
name="images_root"
path="." />
</paths>
我的主要问题,是root-path的问题,谷歌官方只简单的提供了其他几种方式的共享路径,但是对于手机根目录没有提及,所以我在测试华为机器的时候,每次路径都跟其他的机型不一样,其他机型返回的都是正常的:
/storage/emulated/0/...
但是华为却给我返回的是:
/storage/....
这就搞的我好尴尬
然后查阅了好多资料,发现,这是因为,谷歌提及的方式对手机内部存储区文件共享是可以行的通的,但是对于外置SD卡是不行的,会报一个
IllegalArgumentException:Failed to find configured root that contains /storage/...
的错误.
查阅了好多资料,发现可以试着用root-path来解决读取外置sd卡的问题
一个大神看了源码后,发现root-path这个可以解决问题.(谷歌官方没有提及)
root-path代表/也就是Android设备的根目录,该目录下包含着手机内部存储器,外置SD卡等所有文件的目录。
然后在我们需要的时候通过系统的getUriForFile(...)方法,获取一个uri路径即可
File file = new File(imagePath);
Uri photoURI = FileProvider.getUriForFile(activity, "com.nbeebank.licaishi.fileprovider", file);
我的项目里,需要打开相册选择照片和通过拍照获取照片,所以,需要在两个地方,都要这样获取uri
具体的例子,可以参考这两个连接
http://www.jianshu.com/p/3f9e3fc38eae