在Android7.0的系統上調用系統相機拍照或者進相冊選擇圖片時,會報以下錯誤: android.os.FileUriExposedException: ******** exposed beyond app through Intent.getData()html
其實不只是調用相機和相冊,只要是訪問文件,都會出現這個錯誤,其緣由是Android 7.0 作了一些系統權限更改,爲了提升私有文件的安全性,面向 Android 7.0 或更高版本的應用私有目錄被限制訪問,此設置可防止私有文件的元數據泄漏,如它們的大小或存在性。而此權限更改有多重反作用,其中之一就是當傳遞軟件包網域外的 file:// URI 可能給接收器留下沒法訪問的路徑。所以,嘗試傳遞 file:// URI 會觸發 FileUriExposedException。分享私有文件內容的推薦方法是使用 FileProvider。在應用間共享文件 對於面向 Android 7.0 的應用,Android 框架執行的 StrictMode API 政策禁止在您的應用外部公開 file:// URI。若是一項包含文件 URI 的 intent 離開您的應用,則應用出現故障,並出現 FileUriExposedException 異常。要在應用間共享文件,應發送一項 content:// URI,並授予 URI 臨時訪問權限。進行此受權的最簡單方式是使用 FileProvider 類。android
官方的解決辦法Setting Up File Sharing安全
第一步:在AndroidManifest中配置:app
<provider框架
android:name="android.support.v4.content.FileProvider"ide
android:authorities="{程序包名}.provider"ui
android:exported="false"this
android:grantUriPermissions="true">google
<meta-dataspa
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths" />
</provider>
說明:android:authorities="{程序包名}.provider" 這個值能夠隨便寫。我用的是程序包名。這個值決定了fileProVider生成的uri的路徑。後面詳細介紹
exported:要求必須爲false,爲true則會報安全異常。grantUriPermissions:true,表示授予 URI 臨時訪問權限。
第二步:res目錄下新建一個xml文件夾,而後新建文件provider_paths.xml(同配置文件中的resource值一致),其內容爲
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path name="files_root" path="Android/data/{包名}/"/>
<external-path name="external_storage_root" path="."/>
</paths>
第三部程序中使用:
Uri takePhotoUri;
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
String takePhotoPath = Tools.getSDPath(Constants.ImageCameraPath)+ Tools.getPhotoFileName();// 拍照後的照片路徑(本身設置)
File file = new File(takePhotoPath);//建立圖片文件
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
takePhotoUri = FileProvider.getUriForFile(this, getPackageName() +".provider", file);
} else {
takePhotoUri= Uri.fromFile(file);
}
intent.putExtra(MediaStore.EXTRA_OUTPUT, takePhotoUri);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 0);
startActivityForResult(intent, CAMERA_WITH_DATA);
說明:FileProvider.getUriForFile();方法中的第二個參數要跟配置文件中的android:authorities值相對應。
第四步,獲取拍照完成後的圖片
在 onActivityResult()方法中使用data.getData();方法來獲取 Uri實際是行不通的,能夠在第三步的時候把拍照後的照片路徑takePhotoPath設爲一個全局變量
這樣的話就能夠直接使用這個路徑來獲取拍照後的圖片。
String crop_image = new SimpleDateFormat("yyyy_MMdd_hhmmss").format(new Date()) + "_crop" + ".jpg";//剪切後的圖片
File cropFile = createFile(crop_image); File file = new File(takePhotoPath);//直接使用第三步設置的路徑來獲取拍照後的圖片
Uri imageUri;
Uri cropUri;
Intent intent = new Intent("com.android.camera.action.CROP");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
imageUri = FileProvider.getUriForFile(this, getPackageName() +".provider", file); cropUri = Uri.fromFile(cropFile);
} else {
imageUri = Uri.fromFile(file); cropUri = Uri.fromFile(cropFile);
}
intent.setDataAndType(imageUri, "image/*");
intent.putExtra("crop", "true"); //設置寬高比例
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1); //設置裁剪圖片寬高
intent.putExtra("outputX", 400);
intent.putExtra("outputY", 400);
intent.putExtra("scale", true); //裁剪成功之後保存的位置
intent.putExtra(MediaStore.EXTRA_OUTPUT, cropUri);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); startActivityForResult(intent, CROP_RESULT_CODE);