1. 程式人生 > >Android7.0及以上使用帶uri的Intent訪問檔案的問題

Android7.0及以上使用帶uri的Intent訪問檔案的問題

解決 Android N 上 安裝Apk時報錯:android.os.FileUriExposedException: file:///storage/emulated/0/Download/appName-2.3.0.apk exposed beyond app through Intent.getData()

Android N 系統,Android 框架執行的 StrictMode,API 禁止向您的應用外公開 file://URI。 
如果一項包含檔案 URI 的 Intent 離開您的應用,應用會停止執行,並出現 FileUriExposedException異常。官方文件在Android 7.0 行為變更進行了詳細說明

android.os.FileUriExposedException: 
file:///storage/emulated/0/Download/appName-2.3.0.apk exposed beyond app through Intent.getData()

若要在應用間共享檔案,您應傳送一項 content://URI(代替file://URI),並授予 URI 臨時訪問許可權。

FileProvider這個類就是把一個檔案File,轉換為 content://URI的

FileProvider是ContentProvider子類,所以FileProvider的使用方法,和ContentProvider使用基本上是一樣的

解決方法

1、在AndroidManifest.xml中新增如下程式碼
<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="app的包名.fileProvider"
    android:grantUriPermissions="true"
    android:exported="false">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_paths" />
</provider>

注意: 
authorities:app的包名.fileProvider 
grantUriPermissions:必須是true,表示授予 URI 臨時訪問許可權 
exported:必須是false 
resource:中的@xml/file_paths是我們接下來要新增的檔案

2、在res目錄下新建一個xml資料夾,並且新建一個file_paths的xml檔案(如下圖)

這裡寫圖片描述

3、開啟file_paths.xml檔案新增如下內容
<?xml version="1.0" encoding="utf-8"?>
<paths>
    <external-path path="Android/data/app的包名/" name="files_root" />
    <external-path path="." name="external_storage_root" />
</paths>

path:需要臨時授權訪問的路徑(.代表所有路徑) 
name:就是你給這個訪問路徑起個名字

4、修改程式碼適配Android N
Intent intent = new Intent(Intent.ACTION_VIEW);
//判斷是否是AndroidN以及更高的版本
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
    intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    Uri contentUri = FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID + ".fileProvider", apkFile);
    intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
} else {
    intent.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
}
startActivity(intent);

1、首先我們對Android N及以上做判斷; 
2、然後新增flags,表明我們要被授予什麼樣的臨時許可權 
3、以前我們直接 Uri.fromFile(apkFile)構建出一個Uri,現在我們使用FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID + ".fileProvider", apkFile); 
4、BuildConfig.APPLICATION_ID直接是應用的包名

7.0拍照方法:

/**
     * 開啟相機拍照
     *
     * @param activity
     * @return
     */
    public static void openCamera(Activity activity) {

        String filename = new SimpleDateFormat("yyyyMMdd-HHmmss", Locale.CHINA)
                    .format(new Date()) + ".png";

        File pictureFile = new File(Environment.getExternalStorageDirectory(), filename );

        Intent mIntent = new Intent();
        mIntent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

            Uri contentUri = FileProvider.getUriForFile(activity, "app的包名.fileProvider", pictureFile );
            //拍照結果輸出到這個uri對應的file中
            mIntent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
            //對這個uri進行授權
            mIntent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        } else {
            //拍照結果輸出到這個uri對應的file中
            mIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(pictureFile ));
        }

        mIntent.putExtra(MediaStore.Images.Media.ORIENTATION, 0);
        activity.startActivityForResult(mIntent, REQUEST_CAMERA_IMAGE);
    }


核心程式碼就這一行了~

Uri contentUri = FileProvider.getUriForFile(activity, "app的包名.fileProvider", pictureFile );
1
第二個引數就是我們配置的authorities,這個很正常了,總得對映到確定的ContentProvider吧~所以需要這個引數。

第三個引數是指定的檔案File

生成的uri:

content://com.xuexuan.fileprovider/external/20171201-094017.png

參考地址

                    <link rel="stylesheet" href="http://csdnimg.cn/release/phoenix/production/markdown_views-0bc64ada25.css">
                        </div>