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

升级到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);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

其中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);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

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>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

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>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

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;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

关于相机裁剪 
相机裁剪 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
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39

参考文档

[1]Android N 调用相册crash- FileUriExposedException 
[2]根据 Android Training课程写的FileProvider小例子 
[3]使用FileProvider共享文件 
[4]Android7.0适配教程与心得 
[5]Android N 调用相册crash- FileUriExposedException

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值