我个人建议大家做android的一定要多几台主流点的设备测试
我今天又采坑了,调用相机的时候报了个FileUriExposedException
第一反应就是权限问题......
Google反对放宽私有目录的访问权限的,所以收起对私有文件的访问权限是Android将来发展的趋势。
Android7.0之后直接使用本地的根目录即file:// URI是不安全的操作,所以会触发 FileUriExposedException
这就意味着在Android7.0以前我们访问相机拍照存储时,如果使用URI的方式直接存储剪裁图片就会造成这个异常
因此Google为我们提供了FileProvider类,FileProvider时ContentProvide的子类,ContentProvide是Android的四大组件之一,它使用了和内容提供器类似的机制来对数据进行保护,可以选择性地将封装过的Uri共享给外部,从而提高了应用的安全性。
使用FileProvider获取Uri就会将以前的file:// URI准换成content:// URI,实现一种安全的应用间数据访问
使用FileProvider需要在清单文件AndroidManifest.xml中进行注册的:
xxxxxx:我这里用的是自己的项目包名,这个只是个标识
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="xxxxxx.fileProvider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths" />
</provider>
provider标签里的
android:name: FileProvider的包名+类名
android:authorities: 是一个自定义的标识,使用FileProvider的getUriForFile方法时的一个参数和清单文件注册时一致即可
exported: 必须为false,为true则会报安全异常
grantUriPermissions: true,表示授予 URI 临时访问权限
<meta-data />标签里面是用来指定共享的路径
android:resource="@xml/provider_paths"就是我们的共享路径配置的xml文件
在res目录下创建xml文件夹,provider_paths.xml文件内容如下:
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path path="image/" name="image" />
</paths>
效果:
然后就是调用相机的代码了:
/**
* 跳转到照相机
*/
//调用照相机返回图片文件
private File tempFile;
private void gotoCamera() {
Log.d("gotoCamera()", "*****************打开相机********************");
//创建拍照存储的图片文件
tempFile = new File(FileUtil.checkDirPath(Environment.getExternalStorageDirectory().getPath() + "/image/"), System.currentTimeMillis() + ".jpg");
//跳转到调用系统相机
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
//这里区分下android版本再进行不同的操作
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
//设置7.0中共享文件,分享路径定义在xml/provider_pathsaths.xml
intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
//BuildConfig.APPLICATION_ID + ".fileProvider"
//这个参数就是之前说到的android:authorities标识了
Uri contentUri = FileProvider.getUriForFile(UserSetMessageActivity.this,
BuildConfig.APPLICATION_ID + ".fileProvider", tempFile);
intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
} else {
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile));
}
startActivityForResult(intent, CAMERA_WITH_DATA);
}