1. 程式人生 > >Android7.0解決 android.os.FileUriExposedException: file:///storage/emulated/0/

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>