“私有目錄被限制訪問”是指在 Android7.0 中為了提高私有文件的安全性昂秃,面向 Android N 或更高版本的應(yīng)用私有目錄將被限制訪問父阻,會出現(xiàn) android.os.FileUriExposedException
而7.0的” StrictMode API 政策” 是指禁止向你的應(yīng)用外公開 file:// URI。 如果一項(xiàng)包含文件 file:// URI類型 的 Intent 離開你的應(yīng)用侄旬,應(yīng)用失敗肺蔚,并出現(xiàn) FileUriExposedException 異常。之前代碼用到的Uri.fromFile就是商城一個(gè)file://的Uri 在7.0之后,我們需要使用 FileProvider 來解決
第一步:在 AndroidManifest.xml 清單文件中注冊 provider
<!--7.0+ 安裝apk文件-->
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="<包名>.installapk"
android:exported="false"
android:grantUriPermissions="true">
<!--元數(shù)據(jù)-->
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
需要注意一下幾點(diǎn):
- exported:必須為 false
- grantUriPermissions : true儡羔,表示授予 URI 臨時(shí)訪問權(quán)限宣羊。
- authorities 組件標(biāo)識璧诵,都以包名開頭,避免和其它應(yīng)用發(fā)生沖突。
第二步:指定共享文件的目錄,需要在 res文件夾 中新建 xml 目錄,并且創(chuàng)建 file_paths.xml
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android">
<paths>
<external-path
name="download"
path="" />
</paths>
</resources>
path=”“段只,是有特殊意義的腮猖,它代表根目錄,也就是說你可以向其它的應(yīng)用共享根目錄及其子目錄下任何一個(gè)文件了赞枕。
第三步:使用 FileProvider
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
//參數(shù)2 :Provider主機(jī)地址 和 配置文件中保持一致澈缺;參數(shù)3:共享的文件
Uri apkUri = FileProvider.getUriForFile(ctx, "", apkFile);
//對應(yīng)用臨時(shí)授權(quán)該URI代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.setDataAndType(apkUri, INTENT_TYPE);
ctx.startActivity(intent);
混合適配所有版本的代碼:
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Uri uri;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
uri = Uri.fromFile(apkFile);
} else {
//參數(shù)2 :Provider主機(jī)地址 和 配置文件中保持一致;參數(shù)3:共享的文件
uri = FileProvider.getUriForFile(ctx, PROVIDER_AUTHORITY, apkFile);
//對應(yīng)用臨時(shí)授權(quán)該URI代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
intent.setDataAndType(uri, INTENT_TYPE);
ctx.startActivity(intent);