安卓7.0针对应用内部文件权限做了安全性限制,如果外部应用想要读取内部文件,需要通过fileprovider的形式生成临时URI给外部应用
最典型的应用是应用内检查更新,涉及到下载-调用系统安装器(系统安装器相对来说也就是其他应用)安装下载好的APK
首先AndroidManifest.xml里application节点里添加:
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${applicationId}.fileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
然后在XML下增加file_paths.xml(这里我只把升级的apk下载到data/data/包名/files/download/目录下,所以这里只列了这一个目录)
<?xml version="1.0" encoding="utf-8"?>
<resources>
<files-path path="/download/" name="download"/>
</resources>
在这里有可能会遇到Failed to find configured root that contains xxxxx的问题,其实是这个xml里的filepath是有对应关系的,要根据你文件的实际位置来确定。
比如你的文件下载到data/data/包名/cache目录下,那你就应该用这个:
<cache-path name="/download/" path="download" />
xml目录对应path类的目录关系如下:
<files-path path="path" name="name" />
<!-- 对应Context.getFilesDir返回的路径:"/data/data/包名/files"-->
<cache-path name="name" path="path" />
<!-- 对应getCacheDir返回的路径:“/data/data/包名/cache”-->
<external-path name="name" path="path" />
<!-- 对应Environment.getExternalStorageDirectory返回的路径:"/storage/emulated/0"-->
<external-files-path name="name" path="path" />
<!-- 对应Context.getExternalFilesDir(String)/Context.getExternalFilesDir(null)返回的路径:"/storage/emulated/0/Android/data/包名/files"-->
<external-cache-path name="name" path="path" />
<!-- 对应Context.getExternalCacheDir()返回的路径:"/storage/emulated/0/Android/data/包名/cache"-->
FileProvider准备就绪之后就可以创建URI通过Intent传给系统安装器了:
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
if (Build.VERSION.SDK_INT >= 24) {//适配7.0以上系统的FileProvider
Uri apkUri =FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID + ".fileProvider", apkfile);///-----ide文件提供者名
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
}else {
intent.setDataAndType(Uri.fromFile(apkfile),"application/vnd.android.package-archive");
}
context.startActivity(intent);
参考:Failed to find configured root that contains - Arcturis - 博客园