最近作者又碰到因为android 7.0 引起的兼容问题了。
在7.0以前的版本:
//创建临时图片
File photoOutputFile = SDPath.getFile("temp.jpg", SDPath.PHOTO_FILE_STR);
Uri photoOutputUri = Uri.fromFile(photoOutputFile);
这个file文件直接非常简单的转换成"file://XXX/XXX/XXX"的uri格式
7.0后的版本:
当把targetSdkVersion指定成24及之上并且在API>=24的设备上运行时。这种方式则会出现FileUriExposedException异常
android.os.FileUriExposedException: file:///XXX exposed beyond app through ClipData.Item.getUri()
at android.os.StrictMode.onFileUriExposed(StrictMode.java:1799)
at android.net.Uri.checkFileUriExposed(Uri.java:2346)
at android.content.ClipData.prepareToLeaveProcess(ClipData.java:832)
at android.content.Intent.prepareToLeaveProcess(Intent.java:8909)
...
原因
Android不再允许在app中把file://Uri暴露给其他app,包括但不局限于通过Intent或ClipData 等方法。
原因在于使用file://Uri会有一些风险,比如:
- 文件是私有的,接收
file://Uri 的app无法访问该文件。
- 在Android6.0之后引入运行时权限,如果接收file://Uri的app没有申请
READ_EXTERNAL_STORAGE 权限,在读取文件时会引发崩溃。
因此,google提供了FileProvider ,使用它可以生成content://Uri 来替代file://Uri 。
解决方案
首先在AndroidManifest.xml 中添加provider
-
android:authorities 是用来标识provider的唯一标识,在同一部手机上一个"authority"串只能被一个app使用,冲突的话会导致app无法安装。
-
android:exported 必须设置成false ,后面异常会讲为什么
-
android:grantUriPermissions 用来控制共享文件的访问权限,也可以在java代码中设置。
因此,google提供了FileProvider,使用它可以生成content://Uri来替代file://Uri。
解决方案
首先在AndroidManifest.xml中添加provider
android:authorities
是用来标识provider的唯一标识,在同一部手机上一个"authority"串只能被一个app使用,冲突的话会导致app无法安装。
android:exported必须设置成false,后面异常会讲为什么
android:grantUriPermissions用来控制共享文件的访问权限,也可以在java代码中设置。
解决方案
首先在AndroidManifest.xml 中添加provider
-
android:authorities 是用来标识provider的唯一标识,在同一部手机上一个"authority"串只能被一个app使用,冲突的话会导致app无法安装。
-
android:exported 必须设置成false ,后面异常会讲为什么
-
android:grantUriPermissions 用来控制共享文件的访问权限,也可以在java代码中设置。
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.zhongjh.phone.ui"
···
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.zhongjh.phone.ui.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths" />
</provider>
</manifest >
res/xml/provider_paths.xml 这是指定路径和转换规则 <paths> 中可以定义以下子节点
子节点 |
对应路径 |
例子 |
files-path |
Context.getFilesDir() |
cache-path |
Context.getCacheDir() |
external-path |
Environment.getExternalStorageDirectory() |
/storage/emulated/0/ |
external-files-path |
Context.getExternalFilesDir(null) |
external-cache-path |
Context.getExternalCacheDir() |
加入我要替换的目录是 /storage/emulated/0/diary sdcard/photo/ 那么配置应该写成
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="external_files" path="diary sdcard/photo"/>
</paths>
然后修改代码
//创建临时图片
File photoOutputFile = SDPath.getFile("temp.jpg", SDPath.PHOTO_FILE_STR);
//Uri photoOutputUri = Uri.fromFile(photoOutputFile);
Uri photoOutputUri = FileProvider.getUriForFile(
mContext,
mActivity.getPackageName() + ".fileprovider",
photoOutputFile);
intent.putExtra(MediaStore.EXTRA_OUTPUT, photoOutputUri);
我所碰到的异常处理
-
java.lang.SecurityException: Provider must not be exported 解决方案:android:exported 必须设置成false
-
Attempt to invoke virtual method 'android.content.res.XmlResourceParser android.content.pm.PackageItemInfo.loadXmlMetaData(android.content.pm.PackageManager, java.lang.String)' on a null object reference 解决方案:AndroidManifest.xml 处的android:authorities 必须跟mActivity.getPackageName() + ".fileprovider" 一样
|