关于Android 7.0相机FileUriExposedException解决

在开发Android项目的时候,我们会用到相机,有些时候只是开发一个普通的扫码,仅仅赋予一下 权限 就好了,但是有些时候是需要拍照和从相册中获取照片的。

我们在Android 5.0以及5.0之前调用相机可以这样写

Intent intent = new Intent();
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
File savePhoto = new File(Environment.getExternalStorageDirectory().getAbsolutePath(),"/test/"+System.currentTimeMillis() + ".png");
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(savePhoto));
startActivityForResult(intent,200);

 

这样写在6.0之前是完全没有问题的,拍照也可以按照指定的路径进行存储,一切的一切都是OK的,除非部分机型会有问题

到了6.0,在调用相机就得这样写
当我们开发者把sdk升级到了23后,这样写就会存在一点缺陷。那就是权限管理,需要动态进行获取了。OK,然后我们改成动态获取的,如下:

if(checkSelfPermission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){
    requestPermissions(new String[]{Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE},200);
    return;
}
Intent intent = new Intent();
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
File savePhoto = new File(Environment.getExternalStorageDirectory().getAbsolutePath(),"/test/"+System.currentTimeMillis() + ".png");
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(savePhoto));
startActivityForResult(intent,200);

 

这样写和5.0几乎没有区别,只是在前边加了一个权限校验。因为6.0到了动态权限时代

然而到了7.0,一切的一切都不一样了,因为Android 7.0不允许intent带有file://的URI离开自身的应用了,要不然会抛出FileUriExposedException

想要在自己应用和其他应用之间共享File数据,只能使用content://的方式
所以我们在想按照5.0和6.0的方式去调用相机是不可行的了,我们需要在7.0的时候。把所有应用与应用之间的文件传递改成content://的方式,并且还需要把该URI赋予临时的访问权限,使用如下:

 

1.现在清单文件里配置一个provider

<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="应用包名"
    android:exported="false"
    android:grantUriPermissions="true">
</provider>

 

2.在xml目录下创建file_paths文件

<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <files-path name="my_images" path="images/"/>
</paths>

 

在file_paths文件夹内声明的都是需要对外共享的目录,比如上面的配置等于 new File(Context.getFilesDir(),"images") 路径

paths内还可以声明很多种类型的标签,每一种标签都代表了一个路径,如下:

<files-path /> = getFilesDir()
<cache-path /> = getCacheDir()
<external-path /> = Environment.getExternalStorageDirectory()
<external-files-path /> = Context#getExternalFilesDir(String) 或 Context.getExternalFilesDir(null)
<external-cache-path /> = Context.getExternalCacheDir()
<external-media-path /> = Context.getExternalMediaDirs()

 

我们在配置的时候。name的作用就是为了隐藏后边的真实路径,为了安全考虑
而后边的path则是需要共享的路径,用标签所代表的路径加上path上的值,就是完整的路径。

写好path文件后,我们在回到清单文件内继续更新 android.support.v4.content.FileProvider 这个Provider的配置,需要把刚刚的file_paths文件和这个provider关联起来,如下

<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="应用包名"
    android:exported="false"
    android:grantUriPermissions="true">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_paths" />
</provider>

 

3.写完这些配置信息后,我们就可以在应用内直接获取需要共享文件的content://URI了,如下

File imagePath = new File(Context.getFilesDir(), "images");
File newFile = new File(imagePath, "test.jpg");
Uri contentUri = FileProvider.getUriForFile(getContext(), "应用包名", newFile);

 

getUriForFile的第二个参数内也可以不是应用包名,只要和清单文件内的authorities一致即可

这个contentUri的最后结果就是 content://应用包名/my_images/test.jpg

之所以会生成这个uri,相信很多同学看到这里就明白了,因为这个uri是要对其他应用共享的,所以不能直接共享真实路径,便衍生了FileProvider这种东西来专门生成这个Uri,他的生成规则便是 content:// 应用包名(或者是其他字符串)/ 伪名 (path文件内配置的name属性) / 文件名

4.然后赋予临时访问权限

我们可以调用 Context.grantUriPermission 方法来给Uri赋予权限,调用 revokeUriPermission() 函数来撤销权限,也可以通过 Intent.setFlags() 的方式来赋予临时访问权限

最终在7.0上调用相机就成了这个样子,前提是file_paths文件已经配置OK

if(checkSelfPermission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){
    requestPermissions(new String[]{Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE},200);
    return;
}
Intent intent = new Intent();
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
File imagePath = new File(Context.getFilesDir(), "images");
File newFile = new File(imagePath, "test.jpg");
Uri contentUri = FileProvider.getUriForFile(getContext(), "应用包名", newFile);
intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
startActivityForResult(intent,200);

 

到这里就结束了,我也是刚刚踩完相机的坑,所以来一篇博客来记录这个经历,由于该内容全部都是用记事本手敲的,所以或多或少可能会有一些拼写错误之类的,欢迎在评论区指出错误,谢谢!

posted @ 2018-04-03 10:44  柯壮  阅读(8554)  评论(0编辑  收藏  举报