本来今天高高兴兴,想把之前封装好的DownloadManager拿到新项目中使用,但是一个个坑的我难受想哭。
原本的流程是,通过DownloadManager下载Apk到本地,收到了下载完成广播之后安装。
第一坑
--
android.os.FileUriExposedException: file:///storage/emulated/0/xxx exposed beyond app through Intent.getData()
原来是我把targetSdk设置到了24,以至于在将我的文件暴露给其他应用时,抛出了这个异常,下面是使用到的代码。
Intent install = new Intent(Intent.ACTION_VIEW);
install.setDataAndType(apkUri, "application/vnd.android.package-archive");
install.setFlags(FLAG_ACTIVITY_NEW_TASK);
context.startActivity(install);
于是我就按这篇文章所说的,加入了FileProvider,我下载好的文件存放在Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS)下,因此FileProvider的xml是这样设置的。
看下面这个图就清楚了,图片来源,侵删。通过FileProvider.getUriForFile(context, FILE_PROVIDER_NAME, file),获得下载在公共存储Download目录下的apk文件的uri,然后我将这个uri传给安装Intent,应该就没问题了吧。
FileProvider映射图示
图样,第二坑
java.lang.IllegalArgumentException: Failed to find configured root that contains /sdcard/storage/emulated/0/Download/apkFile.apk
在这个问答里,说是xml中的目录写错了,于是改来改去,不得其所。再看到这篇文章,原来是FileProvider不支持sdcard目录下的文件共享。好吧,那我就把apk下载到我的私有目录下,来修改DownloadManager的参数。
public Request setDestinationUri(Uri uri) {
...
}
public Request setDestinationInExternalFilesDir(Context context, String dirType,String subPath) {
...
}
public Request setDestinationInExternalPublicDir(String dirType, String subPath) {
...
}
对,你没看错,指定地址的三个方法,都是存在sd卡上的。咋搞。那我就在下载好了之后,把文件拷贝到我的私有目录下,这总行了吧。
注册一个BroadcastReceiver,然后在onReceive()中,将下载好的存放在sdcard中的apk拷贝到私有目录files文件夹内的downloads目录,下面是BroadcastReceiver的IntentFilter和更改后的FileProvider的xml。
IntentFilter filter = new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE);
测试通过
这里还有点要注意。
Intent install = new Intent(Intent.ACTION_VIEW);
install.setDataAndType(apkUri, "application/vnd.android.package-archive");
// 这里加上 FLAG_GRANT_READ_URI_PERMISSION ,给目标程序读改uri的权限。
install.setFlags(FLAG_GRANT_READ_URI_PERMISSION | FLAG_ACTIVITY_NEW_TASK);
context.startActivity(install);
总结
FileProvider设置的本意是好的,避免直接暴露了应用内部的文件结构。但是提供了external-path获取外部存储空间却不奏效,很是疑惑,不知道是不是我哪里写错了,如果有知道的读者还请指点。