“私有目录被限制访问”是指在 Android7.0 中为了提高私有文件的安全性,面向 Android N 或更高版本的应用私有目录将被限制访问,会出现 android.os.FileUriExposedException
而7.0的” StrictMode API 政策” 是指禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,应用失败,并出现 FileUriExposedException 异常。之前代码用到的Uri.fromFile就是商城一个file://的Uri 在7.0之后,我们需要使用 FileProvider 来解决
第一步:在 AndroidManifest.xml 清单文件中注册 provider
<!--7.0+ 安装apk文件-->
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="<包名>.installapk"
android:exported="false"
android:grantUriPermissions="true">
<!--元数据-->
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
需要注意一下几点:
- exported:必须为 false
- grantUriPermissions : true,表示授予 URI 临时访问权限。
- authorities 组件标识,都以包名开头,避免和其它应用发生冲突。
第二步:指定共享文件的目录,需要在 res文件夹 中新建 xml 目录,并且创建 file_paths.xml
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android">
<paths>
<external-path
name="download"
path="" />
</paths>
</resources>
path=”“,是有特殊意义的,它代表根目录,也就是说你可以向其它的应用共享根目录及其子目录下任何一个文件了。
第三步:使用 FileProvider
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
//参数2 :Provider主机地址 和 配置文件中保持一致;参数3:共享的文件
Uri apkUri = FileProvider.getUriForFile(ctx, "", apkFile);
//对应用临时授权该URI代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.setDataAndType(apkUri, INTENT_TYPE);
ctx.startActivity(intent);
混合适配所有版本的代码:
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Uri uri;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
uri = Uri.fromFile(apkFile);
} else {
//参数2 :Provider主机地址 和 配置文件中保持一致;参数3:共享的文件
uri = FileProvider.getUriForFile(ctx, PROVIDER_AUTHORITY, apkFile);
//对应用临时授权该URI代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
intent.setDataAndType(uri, INTENT_TYPE);
ctx.startActivity(intent);