Android N之前洲炊,通過(guò)Intent安裝一個(gè)APK代碼差多不是下面這個(gè)樣子的:
Intent install = new Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
File apkFile = new File(getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS),"app.apk";
install.setDataAndType(Uri.fromFile(apkFile)), "application/vnd.android.package-archive");
startActivity(install)
上面代碼在Android N(API 26)之前是沒(méi)有問(wèn)題的,但是從Android 7.0開(kāi)始鞠柄,系統(tǒng)修改了安全機(jī)制: 限定應(yīng)用在默認(rèn)情況下只能訪問(wèn)自身應(yīng)用數(shù)據(jù)猜丹。所以當(dāng)我們想通過(guò)File對(duì)象訪問(wèn)其它package數(shù)據(jù)時(shí)鳖昌,就需要借助于ContentProvider卤唉、FileProvider這些組件涩惑,否則會(huì)報(bào)FileUriExposedException異常。
相應(yīng)的桑驱,顯然上面的代碼在Android N中會(huì)有FileUriExposedException的問(wèn)題竭恬,所以我們需要借助FileProvider做出下面的修改。
使用FileProvider的步驟
添加Provider到AndroidManifest.xml
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.android7.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
創(chuàng)建@xml/file_paths對(duì)應(yīng)的資源文件
<?xml version="1.0" encoding="utf-8"?>
<paths>
<!-- 代表外部存儲(chǔ)區(qū)域的根目錄下的文件 Environment.getExternalStorageDirectory() 根目錄 -->
<!-- for: storage/emulated/0/ -->
<external-path
name="external_storage_root"
path="." />
<!-- 代表外部存儲(chǔ)區(qū)域的根目錄下的文件 Environment.getExternalStorageDirectory()/Pictures/ 根目錄 -->
<!-- for: storage/emulated/0/Pictures/ -->
<external-path
name="Pictures"
path="Pictures/" />
<!-- 代表app 私有的存儲(chǔ)區(qū)域 Context.getCacheDir() 根目錄下的images目錄 -->
<!-- for: /data/user/0/com.hm.demo/cache/images -->
<cache-path
name="hm_private_cache"
path="images" />
<!-- 代表app 私有的存儲(chǔ)區(qū)域 Context.getFilesDir() 根目錄下的images目錄 -->
<!-- for: /data/user/0/com.hm.demo/files/images -->
<files-path
name="hm_private_files"
path="images" />
<!-- 代表app 外部存儲(chǔ)區(qū)域根目錄下的文件 Context.getExternalCacheDir 根目錄 -->
<!-- for: /storage/emulated/0/Android/data/com.hm.demo/cache/-->
<external-cache-path
name="external_cache"
path="." />
<!-- 代表app 外部存儲(chǔ)區(qū)域根目錄下的文件 Context.getExternalFilesDir 根目錄 -->
<!-- for: /storage/emulated/0/Android/data/com.hm.demo/files/-->
<external-files-path
name="external_files"
path="." />
</paths>
paths里面用來(lái)放你需要訪問(wèn)的非本應(yīng)用路徑熬的,只有通過(guò)這里申明后痊硕,才能在程序中使用不報(bào)錯(cuò)。
- files-path 對(duì)應(yīng) Context.getFilesDir()
- cache-path 對(duì)應(yīng) getCacheDir()
- external-path 對(duì)應(yīng) Environment.getExternalStorageDirectory()
- external-files-path 對(duì)應(yīng) Context#getExternalFilesDir(String) Context.getExternalFilesDir(null)
- external-cache-path 對(duì)應(yīng) Context.getExternalCacheDir()
具體其中的path和name就需要你自己來(lái)根據(jù)需求確定了押框。
應(yīng)用代碼
Intent install = new Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
File apkFile = new File(getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS),"app.apk";
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
install.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
Uri contentUri = FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID+".android7.fileProvider", apkFile);
install.setDataAndType(contentUri, "application/vnd.android.package-archive");
} else {
install.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
}
startActivity(install);