在 Android 7.0 ( N ) 以后调用调用相册、照片、安装 APK 等等都会有可能遇到异常 android.os.FileUriExposedException
记得之前有处理过这个问题,今天在安装 APK 时,也遇到以为比较简单,可以快速解决。没想到一天都困在这个 bug 上了(后面讲原因)。经过今天大量的搜索资料,算是系统的学习了一遍这个知识点了吧。
一、在 Android 7.0 以上,对于应用间共享文件做了强制要求。
对于面向 Android N 的应用,Android 框架执行的 StrictMode,API 禁止向您的应用外公开 file://URI。如果一项包含文件 URI 的 Intent 离开您的应用,应用失败,并出现 FileUriExposedException异常。
若要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。
二、使用方法
- 在 AndroidManifest.xml 中添加 provider
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.fileprovider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths" />
</provider>
- 创建 XML 文件,如:filepaths.xml
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<root-path name="root" path="" />
<files-path name="files" path="." />
<cache-path name="cache" path="." />
<external-path name="external" path="." />
<external-files-path name="name" path="path" />
<external-cache-path name="name" path="path" />
</paths>
- files-path :Context.getFilesDir()
- cache-path:getCacheDir()
- external-path:Environment.getExternalStorageDirectory()
参考:
<!-- /storage/emulated/0/Download/${applicationId}/.beta/apk-->
<external-path name="beta_external_path" path="Download/"/>
<!--/storage/emulated/0/Android/data/${applicationId}/files/apk/-->
<external-path name="beta_external_files_path" path="Android/data/"/>
其中 path = "." 表示所有的目录。
经过一顿乱写之后,从下午一直到晚上,最后旁边的同事都走完了才发现,原来是在调用跳转的方法和调用跳转的地方不是同一个方法,应该一手封装了一个工具类方便调用,后面维护的同事不知道原本写有了,还把代码复制粘贴原本到那个地方,而且这个还不容易找到。唉,心真累。尤其是修改别人的代码。不过我也粗心,看文档,没看代码导致浪费了很多时间。
Uri 获取需要做判断,我这里是调转安装:
File file = new File(apkPath);
Uri uri;
if (Build.VERSION.SDK_INT >= 24) {
uri = getUriForFile(context, context.getPackageName() + ".fileProvider", file);
context.grantUriPermission(context.getPackageName(), uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
} else {
uri = Uri.fromFile(file);
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION |
Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri,
"application/vnd.android.package-archive");
context.startActivity(intent);
android.os.Process.killProcess(android.os.Process.myPid());
ps:最近在集成某语音 SDK 时,遇到一个小坑。
由于内部直接定义了 authorities 并且是写死的,这时会出现安装异常问题了。
Failure [INSTALL_FAILED_CONFLICTING_PROVIDER: Package couldn't be installed in /data/app/com.xxxxx-E4rQAls5E5bmlUawI12PAg==: Can't install because provider name com.xxx.xx.x.mobilesdk.com.xxx.ting.httpclient (in package com.xxx.xx) is already used by com.xxx.xx.x]
因为系统会检测到手机本身已经有了这个 provider 了。所以命名的时候最好是跟随包名变化。
android:authorities="${applicationId}.fileprovider"
版权声明:本文为jacksinrow原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。