安卓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);