Android 9.+ API 29:/storage/emulated/0/Pictures/myPic.png 打开失败:EACCES(权限被拒绝)

lof*_*fty 9 android-studio

我正在使用 Android Virtual Device (AVD) Nexus 7 (2012) API 29 在 Android Studio 3.4.1 上尝试一个非常基本的相机应用程序的源代码。我在模拟器中用相机拍了一张照片,并试图将其保存为 / storage/emulated/0/Pictures/myPic.png,但有一个异常 /storage/emulated/0/Pictures/myPic.png 打开失败:EACCES(权限被拒绝)。我已经尝试了所有在线提供的建议,例如:

  1. 在 AndroidManifest.xml 中,我添加了:
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

Run Code Online (Sandbox Code Playgroud)
  1. 在 MainActivity.java 中,我添加了以下运行时权限检查和请求代码:
public class MainActivity extends Activity {

    private static final int REQUEST_EXTERNAL_STORAGE = 1;
    private static String[] PERMISSIONS_STORAGE = {
            Manifest.permission.READ_EXTERNAL_STORAGE,
            Manifest.permission.WRITE_EXTERNAL_STORAGE
    };

    public static void verifyStoragePermissions(Activity activity) {
        int permission = ActivityCompat.checkSelfPermission(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE);

        boolean checkPermission = (permission == PackageManager.PERMISSION_GRANTED);

        Toast toastPermission = Toast.makeText(activity,
                "Is permission granted? " + checkPermission,
                Toast.LENGTH_SHORT);
        LinearLayout toastLayoutPermission = (LinearLayout) toastPermission.getView();
        TextView toastTVPermission = (TextView) toastLayoutPermission.getChildAt(0);
        toastTVPermission.setTextSize(30);
        toastPermission.show();


        if (permission != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(
                    activity,
                    PERMISSIONS_STORAGE,
                    REQUEST_EXTERNAL_STORAGE
            );
        }
    }

Run Code Online (Sandbox Code Playgroud)
  1. 然后我在 onCreate 和我想保存图片之前检查并请求权限,如下所示:
protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        verifyStoragePermissions(this);


......


private void saveImageToFile(File file)
    {
        if (mCameraBitmap != null)
        {
            FileOutputStream outStream = null;

            verifyStoragePermissions(this);

            try
            {
                outStream = new FileOutputStream(file);

                ...

            }
            catch (Exception e)
            {
                Toast toastException = Toast.makeText(MainActivity.this,
                        "Exception: " + e.getMessage(),
                        Toast.LENGTH_SHORT);
                LinearLayout toastLayoutException = (LinearLayout) toastException.getView();
                TextView toastTVException = (TextView) toastLayoutException.getChildAt(0);
                toastTVException.setTextSize(30);
                toastException.show();
            }

...

Run Code Online (Sandbox Code Playgroud)
  1. 我还在模拟器中手动检查了存储权限,以确保允许访问存储。

但是,没有任何建议有效!

代码抛出了一个异常

/storage/emulated/0/Pictures/myPic.png 打开失败:EACCES(权限被拒绝)

当它到达线

outStream = new FileOutputStream(file);

Run Code Online (Sandbox Code Playgroud)

有什么想法可以进一步解决这个问题吗?非常感谢。

Sam*_*myT 19

android:requestLegacyExternalStorage="true" 添加到 Android Manifest 对我来说适用于 Android 29

<application
    android:name=".MyApplication"
    android:allowBackup="true"
    android:requestLegacyExternalStorage="true" ...
Run Code Online (Sandbox Code Playgroud)


tro*_*man 8

您的问题是由于在 API 29 中引入了作用域存储。您对该问题有三个解决方案(已经提到了一些,但我认为这有助于澄清和巩固):

  1. 仅在运行 API 28 及更低版本的设备上运行您的应用。

  2. 添加android:requestLegacyExternalStorage="true"到 AndroidManifest.xml 以便您的应用程序可以在 API 29 及更高版本上运行:

    <application android:requestLegacyExternalStorage="true" ...

  3. 为 API 29+ 现代化您的应用程序。首先摆脱 WRITE_EXTERNAL_STORAGE 权限。和替换过时的通话Environment.getExternalStoragePublicDirectory()getExternalFilesDir(),这将节省外部存储,但在一个目录只是你的应用程序可以访问你的形象。

下面是一种使用 MediaStore 将位图保存为 JPG 的方法,它允许其他应用程序访问图像:

private void saveImage(Bitmap bitmap) throws IOException {
    ContentValues contentValues = new ContentValues();
    contentValues.put(MediaStore.MediaColumns.DISPLAY_NAME, "myImage.jpg");
    contentValues.put(MediaStore.MediaColumns.MIME_TYPE, "image/jpeg");
    contentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, Environment.DIRECTORY_PICTURES);

    ContentResolver resolver = getContentResolver();
    Uri uri = resolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, contentValues);
    OutputStream imageOutStream = null;

    try {
        if (uri == null) {
            throw new IOException("Failed to insert MediaStore row");
        }

        imageOutStream = resolver.openOutputStream(uri);
        if (!bitmap.compress(Bitmap.CompressFormat.JPEG, 100, imageOutStream)) {
            throw new IOException("Failed to compress bitmap");
        }
    }
    finally {
        if (imageOutStream != null) {
            imageOutStream.close();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)


lof*_*fty 0

我最终发现问题出在Android API 29上,一旦我降级到API 28,就不再有权限问题了。