android 7.0 读写异常,Android 7.0权限适配:FileUriExposedException异常

今天来聊聊Android 7.0 FileUriExposedException异常,以及它的使用方法和使用场景

一 描述

问题

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

解决方案

要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。进行此授权的最简单方式除了将targetSdkVersion改成24以下,就是使用 FileProvider类

官网对FileProvider描述:

FileProvider是ContentProvider的一个特殊子类,它通过创建内容来实现与应用程序相关联的文件的安全共享:// Uri用于文件,而不是文件:/// Uri。

内容URI允许您使用临时访问权限来授予读取和写入访问权限。当您创建包含内容URI的Intent时,为了将内容URI发送到客户端应用程序,还可以调用Intent.setFlags()来添加权限。只要接收活动的堆栈处于活动状态,客户端应用程序就可以使用这些权限。对于要访问服务的意图,只要服务正在运行,权限就可用。

相比之下,为了控制对文件的访问:/// Uri你必须修改底层文件的文件系统权限。您提供的权限可用于任何应用程序,并在您更改之前保持有效。这种访问水平基本上是不安全的。

内容URI提供的增加文件访问安全级别使FileProvider成为Android安全基础架构的关键部分。

二 如何使用FileProvider

1. 定义FileProvider

由于FileProvider的默认功能,包括内容URI代的文件,你不需要在代码中定义一个子类。我们在manifest中声明provider

...

...

android:name="android.support.v4.content.FileProvider"

android:authorities="包名.fileprovider"

android:exported="false"

android:grantUriPermissions="true">

android:name="android.support.FILE_PROVIDER_PATHS"

android:resource="@xml/file_paths"/>

...

android:name 【固定值】 FileProvider的包名+类名

android:authorities 【自定义】 推荐以包名+”.fileprovider”方式命名,增加辨别性,系统唯一

android:exproted 要求必须为false,为true则会报安全异常

android:grantUriPermissions 是否允许为文件设置临时权限 “true”

android:resource="@xml/file_paths"就是我们的共享路径配置的xml文件

2 . 配置file_paths

FileProvider只能生成你事先指定的 content URI,file_paths配置如下:

name="external"

path=""/>

name="my_images"

path="Android/data/包名/files/Pictures/"/>

name="images"

path="Pictures/"/>

注意: 注: XML文件是你可以指定你要共享的目录的唯一途径,你不能以编程方式添加一个目录,至少配置一个external-path节点

在paths节点内部支持以下几个子节点,分别为:

代表设备的根目录new File("/")

代表该文件files/的应用程序的内部存储区的子目录,等同于context.getFilesDir()

代表应用程序的内部存储区域的缓存子目录的文件,等同于context.getCacheDir()

代表在外部存储区根目录的文件,等同于Environment.getExternalStorageDirectory()

代表应用程序的外部存储区根目录的文件,等同于Context.getExternalFilesDir(String) /Context.getExternalFilesDir(null)

代表应用程序的外部缓存区根目录的文件,等同于Context.getExternalCacheDir()

file_paths用来指定Uri共享和真实路径的映射关系,name属性的值可以自定义,path属性的值表示共享的具体位置,设置为空,就表示共享整个SD卡,也可指定对应的SDcard下的文件目录,根据需求自行定义

3. 获得content uri

使用getUriForFile()将file:// 转换成 content://

Uri fileUri = FileProvider.getUriForFile(this, "包名.fileprovider", file);

4. 临时读写权限授权

需要对接收应用设置读权限或写权限亦或读写均设置:

FLAG_GRANT_READ_URI_PERMISSION:读权限

FLAG_GRANT_WRITE_URI_PERMISSION:写权限

授权方式:

使用Intent.addFlags或setFlags,该方式授权的有效期限,权限截止于该 App 所处的堆栈被销毁自动回收(APP销毁),主要用于针对intent.setData,setDataAndType以及setClipData相关方式传递uri

2 使用grantUriPermission(String toPackage, Uri uri, int modeFlags)来进行授权,该方式授权的有效期限,从授权一刻开始,手动调用 Context.revokeUriPermission() 方法或者设备重启才截止

三 使用场景

a. 相机拍照

Android 7.0之前我们这样拍照,没有什么问题(忽略6.0权限问题):

private static final int REQUEST_TAKE_PHOTO = 0X11;

private Uri imageUri ;

private void takePhoto() {

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

//判断是否有相机应用

if (takePictureIntent.resolveActivity(getActivity().getPackageManager()) != null) {

//获取存储路径 没有则创建

File directory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);

if (!directory.exists()) {

if (!directory.mkdir()) {

return;

}

}

File file = new File(directory.getAbsolutePath(), new SimpleDateFormat("yyyyMMdd-HHmmss", Locale.CHINA)

.format(new Date()) + ".jpeg");

imageUri = Uri.fromFile(file);

takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);

startActivityForResult(takePictureIntent, TAKE_PHOTO);

} else {

ToastUtil.showShort(getString(R.string.TakePhoto_Error));

}

}

@Override

public void onActivityResult(int requestCode, int resultCode, Intent data) {

super.onActivityResult(requestCode, resultCode, data);

if (resultCode == RESULT_OK && requestCode == REQUEST_TAKE_PHOTO) {

// 通知图库更新

getActivity().sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, imageUri ));

}

}

如果我们使用Android 7.0或者以上的原生系统运行,发现应用直接停止运行,如文章开头所说抛出了android.os.FileUriExposedException:

android.os.FileUriExposedException:

file:///storage/emulated/0/Pictures/20170723-201847.jpeg exposed beyond app through ClipData.Item.getUri()

at android.os.StrictMode.onFileUriExposed(StrictMode.java:1799)

at android.net.Uri.checkFileUriExposed(Uri.java:2346)

接下来根据官网的解决办法,如第二步所说配置好 FileProvider,更改拍照方法:

private void takePhoto() {

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

//判断是否有相机应用

if (takePictureIntent.resolveActivity(getActivity().getPackageManager()) != null) {

//获取存储路径 没有则创建

File directory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);

if (!directory.exists()) {

if (!directory.mkdir()) {

return;

}

}

File file = new File(directory.getAbsolutePath(), new SimpleDateFormat("yyyyMMdd-HHmmss", Locale.CHINA)

.format(new Date()) + ".jpeg");

Uri uri = imageUri = Uri.fromFile(file);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

//兼容7.0

uri = FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file);

//添加权限 这一句表示对目标应用临时授权该Uri所代表的文件

takePictureIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

takePictureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

}

takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, uri);

startActivityForResult(takePictureIntent, TAKE_PHOTO);

} else {

ToastUtil.showShort(getString(R.string.TakePhoto_Error));

}

}

添加了版本判断,并使用 FileProvider.getUriForFile()获得content Uri,方法主要更改如下:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

//兼容7.0

uri = FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file);

//添加权限 这一句表示对目标应用临时授权该Uri所代表的文件

takePictureIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

takePictureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

}

当然也可以不用判断版本,直接使用FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file)获得Uri替换Uri.fromFile(file),但是切记需要进行授权和取消授权,否则4.4以下会报Permission Denial

b. 图片裁剪

/**

* @param activity 当前activity

* @param orgUri 剪裁原图的Uri

* @param desUri 剪裁后的图片的Uri

* @param aspectX X方向的比例

* @param aspectY Y方向的比例

* @param width 剪裁图片的宽度

* @param height 剪裁图片高度

* @param requestCode 剪裁图片的请求码

*/

public static void cropImageUri(Activity activity, Uri orgUri, Uri desUri, int aspectX, int aspectY, int width, int height, int requestCode) {

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 | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

}

intent.setDataAndType(orgUri, "image/*");

intent.putExtra("crop", "true");

intent.putExtra("aspectX", aspectX);

intent.putExtra("aspectY", aspectY);

intent.putExtra("outputX", width);

intent.putExtra("outputY", height);

intent.putExtra("scale", true);

//将剪切的图片保存到目标Uri中

intent.putExtra(MediaStore.EXTRA_OUTPUT, desUri);

intent.putExtra("return-data", false);

intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());

intent.putExtra("noFaceDetection", true);

activity.startActivityForResult(intent, requestCode);

}

c. 安装apk

// 安装Apk

public void installApk(Context context) {

File file = new File(Environment.getExternalStorageDirectory(), "app.apk");

Intent intent = new Intent(Intent.ACTION_VIEW);

Uri uri = Uri.fromFile(file);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

uri = FileProvider.getUriForFile(context, "包名.fileprovider", file);

intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

}

intent.setDataAndType(uri, "application/vnd.android.package-archive");

context.startActivity(intent);

}

大概使用就这么多,望多多指教。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值