android.os.FileUriExposedException: file://xxx

今天遇到一个下载安装apk失败是因为

Google反对放宽私有目录的访问权限的,所以收起对私有文件的访问权限是Android将来发展的趋势。
Android7.0之后直接使用本地的根目录即file:// URI是不安全的操作,所以会触发 FileUriExposedException
这就意味着在Android7.0以前我们访问相机拍照存储时,如果使用URI的方式直接存储剪裁图片就会造成这个异常

因此Google为我们提供了FileProvider类,FileProvider时ContentProvide的子类,ContentProvide是Android的四大组件之一,它使用了和内容提供器类似的机制来对数据进行保护,可以选择性地将封装过的Uri共享给外部,从而提高了应用的安全性。

使用FileProvider获取Uri就会将以前的file:// URI准换成content:// URI,实现一种安全的应用间数据访问
使用FileProvider需要在清单文件AndroidManifest.xml中进行注册的:

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

provider标签里的
android:name: FileProvider的包名+类名
android:authorities: 是一个自定义的标识,使用FileProvider的getUriForFile方法时的一个参数和清单文件注册时一致即可
exported: 必须为false,为true则会报安全异常
grantUriPermissions: true,表示授予 URI 临时访问权限
标签里面是用来指定共享的路径
android:resource="@xml/apk_provider_paths"就是我们的共享路径配置的xml文件
在这里插入图片描述
写上你的app名称和下载路劲

//安装Apk文件

private void installApk(String path) {
    Intent intent = new Intent(Intent.ACTION_VIEW);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    Uri apkUri ;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        // 授予文件操作的临时权限,根据需求设定,一般安装只需要READ权限
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        // 获取配置的FileProvider的 Content Uri的值
        apkUri = FileProvider.getUriForFile(getApplicationContext()
                , "你的包名.FileProvider"
                , new File(path));
    } else {
        apkUri = Uri.fromFile(new File(path));
    }
    intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
    startActivity(intent);

}

猜你喜欢

转载自blog.csdn.net/xieyaofeng/article/details/103935333