修复安卓N安装APP报FileUriExposedException错误

FileUriExposedException的错误的日志

android.os.FileUriExposedException: file:///storage/emulated/0/MIUI%20Purify/backupAPP/com.tencent.mobileqq.apk exposed beyond app through Intent.getData()
	at android.os.StrictMode.onFileUriExposed(StrictMode.java:1799)
	at android.net.Uri.checkFileUriExposed(Uri.java:2346)
	at android.content.Intent.prepareToLeaveProcess(Intent.java:8988)


	at android.content.Intent.prepareToLeaveProcess(Intent.java:8949)
	at android.app.Instrumentation.execStartActivity(Instrumentation.java:1519)
	at android.app.Activity.startActivityForResult(Activity.java:4225)
	at android.support.v4.app.BaseFragmentActivityJB.startActivityForResult(BaseFragmentActivityJB.java:48)
	at android.support.v4.app.FragmentActivity.startActivityForResult(FragmentActivity.java:77)
	at android.support.v4.app.ActivityCompatJB.startActivityForResult(ActivityCompatJB.java:26)
	at android.support.v4.app.ActivityCompat.startActivityForResult(ActivityCompat.java:146)

报错的代码

Intent intent = new Intent();
 intent.setAction(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(new File(path_backup + appInfo.getPackageName() + ".apk")), "application/vnd.android.package-archive");
startActivity(intent);

说明

这个问题只会在android N上出现,主要是因为N系统更新了更加严格的安全机制导致的。
默认情况下,app都不能访问本应用之外的私有数据。
MODE_WORLD_READABLE MODE_WORLD_WRITEABLE之类的权限设定都将会引发SecurityException异常。
Android 框架强制执行了 StrictMode API 政策禁止向你的应用外公开 file:// URI
当你跨package域传递file://的URI时,接收者得到的将是一个无权访问的路径,因此,这将会触发FileUriExposedException
这项权限的变更将意味着你无法通过File API访问手机存储上的数据了,基于File API的一些文件浏览器等也将受到很大的影响
官方推荐我们可以通过FileProvider或者ContentProvider传递URI

解决

AndroidManifest.xml中添加一个provider

<provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="APP包名.fileprovider"
            android:grantUriPermissions="true"
            android:exported="false"
            >
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths" />
<?/provider>

添加一个XML文件 名为 file_paths.xml

<xml version="1.0" encoding="utf-8"?>
<paths>
    <external-path path="MIUI Purify/backupAPP/" name="files_root" />
    <external-path path="." name="external_storage_root" />
</paths>

"MIUI Purify/backupAPP/"改成你需要的路径

然后调用安装的时候 判断下sdk版本 做区分对待

 Intent intent = new Intent(Intent.ACTION_VIEW);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            Uri contentUri = FileProvider.getUriForFile(context, "com.coderstory.Purify.fileprovider", new File(filePath));
            intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
        } else {
            intent.setDataAndType(Uri.fromFile(new File(filePath)), "application/vnd.android.package-archive");
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        }
        startActivity(intent);