Android 7.0及其以上系統拍照,打開相冊,裁剪,報錯: android.os.FileUriExposedException

在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);

相關文章
相關標籤/搜索