Android7.0及以上使用帶uri的Intent訪問檔案的問題
阿新 • • 發佈:2019-01-26
解決 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>