您的位置:首页 > 移动开发 > Android开发

快速使用FileProvider解决Android7.0文件权限问题

2016-12-12 20:16 513 查看
升级到Android7.0之后,启动系统相机或者截图,传入URI的时候可能会导致程序闪退崩溃。这是因为7.0的新的文件权限导致的。下面是解决这个问题的快速解决方案。

问题代码

在7.0可能会出问题的代码:

final String CACHE_IMG = Environment.getExternalStorageDirectory()+"/demo/"
final int TAG_PHOTO_CAMERA=200;

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

String fileName = "defaultImage.jpg";

File file = new File(CACHE_IMG, fileName);

Uri uri = Uri.fromFile(file);

intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);

startActivityForResult(intent, TAG_PHOTO_CAMERA);


其中Uri uri = Uri.fromFile(file);这里会导致闪退。

解决方法

step1. 将Uri的生成方式改为由FileProvider提供的临时授权路径,并且在intent中添加flag

修改后代码如下

final String CACHE_IMG = Environment.getExternalStorageDirectory()+"/demo/"
final int TAG_PHOTO_CAMERA=200;

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

String fileName = "defaultImage.jpg";

File file = new File(CACHE_IMG, fileName);

Uri imageUri=FileProvider.getUriForFile(activity,"me.xifengwanzhao.fileprovider", file);//这里进行替换uri的获得方式

intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);

intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);//这里加入flag

startActivityForResult(intent, TAG_PHOTO_CAMERA);


step2.在AndroidManifest.xml中的application标签中添加provider的配置

<application
...>
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="me.xifengwanzhao.fileprovider"//这里需要和上面部分字符串相同
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
</application>


step3.在res/xml中新建一个文件file_paths.xml

<?xml version="1.0" encoding="utf-8"?>
<resource xmlns:android="http://schemas.android.com/apk/res/android">
<external-path
name="images"
path="demo/" />
</resource>


OK,大功告成,这样就不会崩溃了

代码解释

我们先看Google官方的7.0行为变更介绍 (不需要翻墙)

对于面向 Android 7.0 的应用,Android 框架执行的 StrictMode API 政策禁止在您的应用外部公开 file:// URI。如果一项包含文件 URI 的 intent 离开您的应用,则应用出现故障,并出现 FileUriExposedException 异常。

要在应用间共享文件,您应发送一项 content:// URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。如需了解有关权限和共享文件的详细信息,请参阅共享文件。

根据文档提示我们使用FileProvider进行处理,同时利用xml对FileProvider进行配置

参考如下

java根路径产生方式对应xml根节点名称
Context.getFilesDir()files-path
getCacheDir()cache-path
Environment.getExternalStorageDirectory()external-path
Context#getExternalFilesDir(String) Context.getExternalFilesDir(null)external-files-path
Context.getExternalCacheDir()external-cache-path
节点中的name 不可重名,path为自定义

关于相册选图和相机裁剪

有同学反映相册选图和相机裁剪时候的报错问题,这里也说一下

系统相册选图返回的Uri是可以直接使用的,不需要也不能使用FileProvider进行转换

如果需要根据uri获得转换后的uri 可以参考如下方式

Uri fromUri;
if (uri.getScheme() != null && uri.getScheme().startsWith("file")) {
fromUri =
FileProvider.getUriForFile(mContext,"me.xifengwanzhao.fileprovider", new File(FileUtils.getPath(mContext, uri)));//这里进行替换uri的获得方式
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);//这里加入flag
} else {
//相册选图适配
fromUri = uri;
}


关于相机裁剪

相机裁剪 intent.setDataAndType(fromUri, “image/*”);这里是需要对uri进行转换的,

而 intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));这里使用原来的方式获取uri就可以了

那么启动系统裁剪的方法可以写成这样

/**
* 开启截图,启动系统的截图方法 返回requestCode为 {Constant.IMG_ZOOM}
*
* @param mContext 必须为activity
* @param uri      需要进行裁剪的图片的uri
* @param size     截图的大小宽和高的数值,这里仅限截图为1:1的正方形
* @return path 截图返回的路径
* @see Constant#IMG_ZOOM
*/
public static String startPhotoZoom(Activity mContext, Uri uri, int size) {
//这里生成一个保存截图用的临时路径并且返回出去
String imgPath;
File file = new File(Constant.ZOOM_IMAGE, Constant.getNewestImageName(mContext));
imgPath = file.getPath();

Intent intent = new Intent("com.android.camera.action.CROP");
Uri fromUri;
if (uri.getScheme() != null && uri.getScheme().startsWith("file")) {
fromUri = FileProvider.getUriForFile(mContext, "me.xifengwanzhao.fileprovider", new File(FileUtils.getPath(mContext, uri)));//这里进行替换uri的获得方式
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);//这里加入flag
} else {
//相册选图适配
fromUri = uri;
}
intent.setDataAndType(fromUri, "image/*");
intent.putExtra("crop", "true");
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));
if (android.os.Build.MANUFACTURER.contains("HUAWEI")) {// 华为特殊处理 不然会显示圆
intent.putExtra("aspectX", 9998);
intent.putExtra("aspectY", 9999);
} else {
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
}
intent.putExtra("outputX", size);
intent.putExtra("outputY", size);
mContext.startActivityForResult(intent, Constant.IMG_ZOOM);
return imgPath;
}


参考文档

[1]Android N 调用相册crash- FileUriExposedException

[2]根据 Android Training课程写的FileProvider小例子

[3]使用FileProvider共享文件

[4]Android7.0适配教程与心得

[5]Android N 调用相册crash- FileUriExposedException
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  android AndroidN
相关文章推荐