在Android7.0的系统上调用系统相机拍照或者进相册选择图片时,会报如下错误: android.os.FileUriExposedException: ******** exposed beyond app through Intent.getData()
产生原因
其实不仅是调用相机和相册,只要是访问文件,都会出现这个错误,其原因是Android 7.0 做了一些系统权限更改,为了提高私有文件的安全性,面向 Android 7.0 或更高版本的应用私有目录被限制访问,此设置可防止私有文件的元数据泄漏,如它们的大小或存在性。而此权限更改有多重副作用,其中之一就是当传递软件包网域外的 file:// URI 可能给接收器留下无法访问的路径。因此,尝试传递 file:// URI 会触发 FileUriExposedException。分享私有文件内容的推荐方法是使用 FileProvider。在应用间共享文件 对于面向 Android 7.0 的应用,Android 框架执行的 StrictMode API 政策禁止在您的应用外部公开 file:// URI。如果一项包含文件 URI 的 intent 离开您的应用,则应用出现故障,并出现 FileUriExposedException 异常。要在应用间共享文件,应发送一项 content:// URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。
解决办法
第一步:在AndroidManifest中配置:
android:name="android.support.v4.content.FileProvider"
android:authorities="{程序包名}.provider"
android:exported="false"
android:grantUriPermissions="true">
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths" />
说明:android:authorities="{程序包名}.provider" 这个值可以随便写。我用的是程序包名。这个值决定了fileProVider生成的uri的路径。后面详细介绍
exported:要求必须为false,为true则会报安全异常。grantUriPermissions:true,表示授予 URI 临时访问权限。
第二步:res目录下新建一个xml文件夹,然后新建文件provider_paths.xml(同配置文件中的resource值一致),其内容为
第三部程序中使用:
Uri takePhotoUri;
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
String takePhotoPath = Tools.getSDPath(Constants.ImageCameraPath)+ Tools.getPhotoFileName();// 拍照后的照片路径(自己设置)
File file = new File(takePhotoPath);//创建图片文件
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
takePhotoUri = FileProvider.getUriForFile(this, getPackageName() +".provider", file);
} else {
takePhotoUri= Uri.fromFile(file);
}
intent.putExtra(MediaStore.EXTRA_OUTPUT, takePhotoUri);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 0);
startActivityForResult(intent, CAMERA_WITH_DATA);
说明:FileProvider.getUriForFile();方法中的第二个参数要跟配置文件中的android:authorities值相对应。
第四步,获取拍照完成后的图片
在 onActivityResult()方法中使用data.getData();方法来获取 Uri实际是行不通的,可以在第三步的时候把拍照后的照片路径takePhotoPath设为一个全局变量
这样的话就可以直接使用这个路径来获取拍照后的图片。
String crop_image = new SimpleDateFormat("yyyy_MMdd_hhmmss").format(new Date()) + "_crop" + ".jpg";//剪切后的图片
File cropFile = createFile(crop_image); File file = new File(takePhotoPath);//直接使用第三步设置的路径来获取拍照后的图片
Uri imageUri;
Uri cropUri;
Intent intent = new Intent("com.android.camera.action.CROP");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
imageUri = FileProvider.getUriForFile(this, getPackageName() +".provider", file); cropUri = Uri.fromFile(cropFile);
} else {
imageUri = Uri.fromFile(file); cropUri = Uri.fromFile(cropFile);
}
intent.setDataAndType(imageUri, "image/*");
intent.putExtra("crop", "true"); //设置宽高比例
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1); //设置裁剪图片宽高
intent.putExtra("outputX", 400);
intent.putExtra("outputY", 400);
intent.putExtra("scale", true); //裁剪成功以后保存的位置
intent.putExtra(MediaStore.EXTRA_OUTPUT, cropUri);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); startActivityForResult(intent, CROP_RESULT_CODE);