Ошибка прав пользователя Android M - ПРОЧИТАЙТЕ И НАПИСАТЬ

Я только что начал с android M, и я не могу получить доступ к внешнему хранилищу. Я получаю следующую ошибку:

Caused by: java.lang.SecurityException: Permission Denial: reading 
com.android.providers.media.MediaProvider uri 
content://media/external/images/media from pid=15355, uid=10053 requires 
android.permission.READ_EXTERNAL_STORAGE, or grantUriPermission() 

Я добавляю разрешение пользователя в манифест, например

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

и мой файл сборки имеет следующие настройки:

compileSdkVersion 23
buildToolsVersion "22.0.1"

minSdkVersion 16
targetSdkVersion 23

Как читать и писать с внешнего хранилища в android M?

Ответ 1

Чтение из документации. Я должен спросить у пользователя разрешения во время выполнения. Пример кода:

Добавить разрешение для манифеста android, как раньше:

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

Проверьте, предоставил ли пользователь разрешение. Если да, пропустите запрос на разрешение и продолжите свой рабочий процесс, попросите у пользователя разрешения:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
    if (!Settings.System.canWrite(this)) {
        requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE,
            Manifest.permission.READ_EXTERNAL_STORAGE}, 2909);
    } else {
        // continue with your code
    }
} else {
    // continue with your code
}

Теперь, чтобы проверить, предоставил ли пользователь разрешение или отказал ему @Override OnRequestPermissionResult, чтобы получить обратный вызов:

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
    switch (requestCode) {
        case 2909: {
            if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                Log.e("Permission", "Granted");
            } else {
                Log.e("Permission", "Denied");
            }
            return;
        }
    }
}

Я не смог ПРОЧИТАТЬ внешнее хранилище, только попросив WRITE разрешения, поэтому я должен был просить Manifest.permission.READ_EXTERNAL_STORAGE.


Также, если вы хотите настроить таргетинг версий ниже api 23, установите флажок Build VERSION во время выполнения с инструкцией IF и запрашивать разрешения только в том случае, если ВЕРСИЯ равна или выше Android M.

Ответ 2

В случае, если вы ищете способ обработки в любой версии Android (вы компилируете для M, но некоторые устройства могут использовать другие версии), есть версия совместимости.

if (ContextCompat.checkSelfPermission(CreatePlayerActivity.this,
            Manifest.permission.READ_EXTERNAL_STORAGE)
            != PackageManager.PERMISSION_GRANTED) {

        // Should we show an explanation?
        if (ActivityCompat.shouldShowRequestPermissionRationale(CreatePlayerActivity.this,
                Manifest.permission.READ_EXTERNAL_STORAGE)) {

            // Show an expanation to the user *asynchronously* -- don't block
            // this thread waiting for the user response! After the user
            // sees the explanation, try again to request the permission.

        } else {

            // No explanation needed, we can request the permission.

            ActivityCompat.requestPermissions(CreatePlayerActivity.this,
                    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
                    MY_PERMISSIONS_REQUEST_READ_CONTACTS);

            // MY_PERMISSIONS_REQUEST_READ_CONTACTS is an
            // app-defined int constant. The callback method gets the
            // result of the request.
        }
    }

Ответ 3

Что касается выбранного ответа:

Settings.System.canWrite(this) и WRITE_EXTERNAL_STORAGE - две разные вещи! Должно быть:

    boolean needsRead = ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
            != PackageManager.PERMISSION_GRANTED;

    boolean needsWrite = ActivityCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)
            != PackageManager.PERMISSION_GRANTED;

    if (needsRead || needsWrite)
        requestStoragePermission()

Ответ 4

Я боролся с разрешением WRITE_EXTERNAL_STORAGE: после того, как я его попросил, диалоговое окно не появилось и сразу же вернулось DENIED.

Проблема была в одной из включенных библиотек. Он содержал одно и то же объявление разрешения в AndroidManifest.xml, но с maxSdkVersion="22". Объявление в файле манифеста приложения было переопределено библиотекой.

Чтобы переопределить разрешение из библиотеки, я использовал следующее:

  <uses-permission
    android:name="android.permission.WRITE_EXTERNAL_STORAGE"
    tools:node="remove"
    tools:selector="com.example.lib" />

Я надеюсь, что это может быть полезно.

Ответ 5

Я использовал эту утилиту для github. Он работает как с API 22, так и с 23. Добавление разрешений во время выполнения не сложно, но для разделения кода и перемещения методов для захвата обратных вызовов немного больно. Эта библиотека обеспечивает привязку api, чтобы сделать все, что вам нужно для поддержки разрешений времени выполнения.

https://github.com/kayvannj/PermissionUtil