Android7.0解決 android.os.FileUriExposedException: file:///storage/emulated/0/
解決Android N檔案訪問crash android.os.FileUriExposedException file:///storage/emulated/0/xxx
原因:
Android N對訪問檔案許可權收回,按照Android N的要求,若要在應用間共享檔案,您應傳送一項 content://URI,並授予 URI 臨時訪問許可權。
而進行此授權的最簡單方式是使用 FileProvider類。
1.在mainfest中加入FileProvider註冊
<application>
<provider
android:authorities="你的應用名.fileprovider" android:name="android.support.v4.content.FileProvider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths"/>
</provider>
</application >
2.配置filepaths檔案
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path path="yang/" name="files_path" />
</paths>
其中:
files-path代表的根目錄: Context.getFilesDir()
external-path代表的根目錄: Environment.getExternalStorageDirectory()
cache-path代表的根目錄: getCacheDir()
<external-path path="honjane/" name="files_path" />
path 代表要共享的目錄
name 只是一個標示,隨便取吧 自己看的懂就ok
for example:通過provider獲取到的uri連結
content://com.ys.providerdemo.fileprovider/files_path/files/b7d4b092822.pdf
name對應到連結中的files_path
path對應到連結中的 files ,當然files是在ys/目錄下
3.訪問檔案
/**
* 開啟檔案
* 當手機中沒有一個app可以開啟file時會拋ActivityNotFoundException
* @param context activity
* @param file File
* @param contentType 檔案型別如:文字(text/html)
*/
public static void startActionFile(Context context, File file, String contentType) throws ActivityNotFoundException {
if (context == null) {
return;
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addCategory(Intent.CATEGORY_DEFAULT);
intent.setDataAndType(getUriForFile(context, file), contentType);
if (!(context instanceof Activity)) {
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
}
context.startActivity(intent);
}
/**
* 開啟相機
*
* @param activity Activity
* @param file File
* @param requestCode result requestCode
*/
public static void startActionCapture(Activity activity, File file, int requestCode) {
if (activity == null) {
return;
}
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_OUTPUT, getUriForFile(activity, file));
activity.startActivityForResult(intent, requestCode);
}
private static Uri getUriForFile(Context context, File file) {
if (context == null || file == null) {
throw new NullPointerException();
}
Uri uri;
if (Build.VERSION.SDK_INT >= 24) {
uri = FileProvider.getUriForFile(context.getApplicationContext(), "你的應用名.fileprovider", file);
} else {
uri = Uri.fromFile(file);
}
return uri;
}
同樣訪問相機相簿都通過FileProvider.getUriForFile申請臨時共享空間
已寫成工具類上傳到github,需要直接下載
使用方法簡單,一行程式碼搞定
開啟檔案:
開啟檔案:
try {
FileUtils.startActionFile(this,path,mContentType);
}catch (ActivityNotFoundException e){
}
呼叫相機:
FileUtils.startActionCapture(this, file, requestCode);
修復bug:
Java.lang.IllegalArgumentException: Failed to find configured root that contains /storage/emulated/0/xxx/xxx/file/12b31d2cab6ed.pdf
external與storage/emulated/0/對應,乍一看貌似沒什麼問題,path設定的是external的根路徑,對應Environment.getExternalStorageDirectory(),
然而這個方法所獲取的只是內建SD卡的路徑,所以當選擇的相簿中的圖片是外接SD卡的時候,就查詢不到圖片地址了,因此便丟擲了failed to find configured root that contains的錯誤。
通過分析FileProvider原始碼發現,在xml解析到對應的標籤後,會執行 buildPath() 方法來將根標籤(files-path,cache-path,external-path等)對應的路徑作為檔案根路徑,
在buildPath(),會根據一些常量判斷是構建哪個目錄下的path,除了上面介紹的幾種path外還有個TAG_ROOT_PATH = “root-path” ,只有當不是root-path時才會去構建其他path,
官方也沒介紹這個root-path,測試了一下發現對應的是DEVICE_ROOT指向的整個儲存的根路徑,這個bug就修復了
修改filepaths檔案:
<paths>
<root-path name="honjane" path="" />
</paths>