Дозвіл READ_EXTERNAL_STORAGE для Android


84

Я намагаюся отримати доступ до мультимедійних файлів (музики) на пристрої користувачів для їх відтворення; простий додаток для музичного програвача "привіт світ".

Я дотримувався деяких підручників, і вони в основному дають однаковий код. Але це не спрацює; він продовжує розбиватися і каже мені:

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

Зараз це мій файл маніфесту:

<?xml version="1.0" encoding="utf-8"?>
<manifest
    xmlns:android="http://schemas.android.com/apk/res/android"
    package="slimsimapps.troff" >

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


<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name=".MainActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
</application>
</manifest>

Це мій метод Java:

public void initialize() {
    ContentResolver contentResolver = getContentResolver();
    Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
    Cursor cursor = contentResolver.query(uri, null, null, null, null);
    if (cursor == null) {
        // query failed, handle error.
    } else if (!cursor.moveToFirst()) {
        // no media on the device
    } else {
        do {
            addSongToXML(cursor);
        } while (cursor.moveToNext());
    }
}

Я намагався:

Щоб розмістити це в різних місцях файлу маніфесту:

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

Щоб додати android: maxSdkVersion при читанні дозволу зовнішнього сховища:

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

Щоб помістити це в маніфест / application / activity-tag:

android:exported=“true

Щоб поставити grantUriPremission між uri та cursro у javamethod:

grantUriPermission(null, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION);

Щоб використати це, воно не вийде з ладу, але курсор стає нульовим:

uri = MediaStore.Audio.Media.getContentUri("EXTERNAL_CONTENT_URI”);

Для використання внутрішнього вмісту uri це працює, як очікувалося, але воно видає лише «звуки ОС», такі як звук затвора, звук низького заряду батареї, клацання кнопки та таке:

uri = MediaStore.Audio.Media.INTERNAL_CONTENT_URI;

Прохання допомогти, це не повинна бути важкою проблемою, я знаю, але я відчуваю себе таким початківцем!

Я прочитав і спробував (або вважав, що вони не стосуються моєї проблеми):

Трасування стека:

09-08 06:59:36.619    2009-2009/slimsimapps.troff D/AndroidRuntime﹕ Shutting down VM
    --------- beginning of crash
09-08 06:59:36.619    2009-2009/slimsimapps.troff E/AndroidRuntime﹕ FATAL EXCEPTION: main
    Process: slimsimapps.troff, PID: 2009
    java.lang.IllegalStateException: Could not execute method for android:onClick
            at android.view.View$DeclaredOnClickListener.onClick(View.java:4452)
            at android.view.View.performClick(View.java:5198)
            at android.view.View$PerformClick.run(View.java:21147)
            at android.os.Handler.handleCallback(Handler.java:739)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:148)
            at android.app.ActivityThread.main(ActivityThread.java:5417)
            at java.lang.reflect.Method.invoke(Native Method)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
     Caused by: java.lang.reflect.InvocationTargetException
            at java.lang.reflect.Method.invoke(Native Method)
            at android.view.View$DeclaredOnClickListener.onClick(View.java:4447)
            at android.view.View.performClick(View.java:5198)
            at android.view.View$PerformClick.run(View.java:21147)
            at android.os.Handler.handleCallback(Handler.java:739)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:148)
            at android.app.ActivityThread.main(ActivityThread.java:5417)
            at java.lang.reflect.Method.invoke(Native Method)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
     Caused by: java.lang.SecurityException: Permission Denial: reading com.android.providers.media.MediaProvider uri content://media/external/audio/media from pid=2009, uid=10059 requires android.permission.READ_EXTERNAL_STORAGE, or grantUriPermission()
            at android.os.Parcel.readException(Parcel.java:1599)
            at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:183)
            at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:135)
            at android.content.ContentProviderProxy.query(ContentProviderNative.java:421)
            at android.content.ContentResolver.query(ContentResolver.java:491)
            at android.content.ContentResolver.query(ContentResolver.java:434)
            at slimsimapps.troff.MainActivity.initialize(MainActivity.java:106)
            at slimsimapps.troff.MainActivity.InitializeExternal(MainActivity.java:80)
            at java.lang.reflect.Method.invoke(Native Method)
            at android.view.View$DeclaredOnClickListener.onClick(View.java:4447)
            at android.view.View.performClick(View.java:5198)
            at android.view.View$PerformClick.run(View.java:21147)
            at android.os.Handler.handleCallback(Handler.java:739)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:148)
            at android.app.ActivityThread.main(ActivityThread.java:5417)
            at java.lang.reflect.Method.invoke(Native Method)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
    --------- beginning of system

Дивно, що помилка відображається, коли ви не вказали цей рядок у маніфесті <koristi-дозвіл android: name = "android.permission.READ_EXTERNAL_STORAGE" /> чи намагаєтесь ви прочитати деякі файли, захищені іншими програмами?
суніл сонячно

На якому рівні api ви це намагаєтесь?
Sharp Edge

Ви пробували чистити проект. ?
суніл сонячно

@sunilsunny Я не намагаюся читати сом файли, які захищені, не те, що я знаю, так чи інакше, просто простий медіаплеєр. Так, я намагався його очистити, я намагався перезавантажити комп’ютер, я спробував створити підписаний файл .apk і опублікувати його в Google Play і отримати до нього доступ як тестер, не маючи удачі ....
Slim Sim

@Гострий край ; Мій AVD - це стандартний Nexus 5, api 23. У моєму модулі gradle є: compileSdkVersion 23 buildToolsVersion "23.0.0" minSdkVersion 14 targetSdkVersion 23 Так би я сказав 23.
Slim Sim

Відповіді:


90

У вас є два варіанти вирішення вашої проблеми. Швидкий - знизити targetApi до 22 (файл build.gradle). По-друге, це використання нової чудової моделі запиту дозволу:

if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE)
        != PackageManager.PERMISSION_GRANTED) {

    // Should we show an explanation?
    if (shouldShowRequestPermissionRationale(
            Manifest.permission.READ_EXTERNAL_STORAGE)) {
        // Explain to the user why we need to read the contacts
    }

    requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
            MY_PERMISSIONS_REQUEST_READ_EXTERNAL_STORAGE);

    // MY_PERMISSIONS_REQUEST_READ_EXTERNAL_STORAGE is an
    // app-defined int constant that should be quite unique

    return;
}

Фрагмент знайдено тут: https://developer.android.com/training/permissions/requesting.html

Рішення 2: Якщо це не спрацює, спробуйте це:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
    && ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
        REQUEST_PERMISSION);

return;

}

а потім у зворотному дзвінку

@Override
public void onRequestPermissionsResult(final int requestCode, @NonNull final String[] permissions, @NonNull final int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_PERMISSION) {
    if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
        // Permission granted.
    } else {
        // User refused to grant permission.
    }
}
}

тобто з коментарів. Дякую


1
Так, звісно! Нову модель дозволу, яку Google представив з останньою версією Android! Дякую, я тестую це якомога швидше і позначу як правильне, якщо це спрацює! (і я починав думати, що про це забули!)
Slim Sim

1
Дозвіл, який вимагається, повинен бути READ_EXTERNAL_STORAGE. Як у цій відповіді .
maclir

swap! = PackageManager.READ_EXTERNAL_STORAGE for! = PackageManager.PERMISSION_GRANTED
Renascienza

1
@Renascienza, так ти маєш рацію. Хтось відредагував цю відповідь неправильно.
piotrpo

2
Що таке MY_PERMISSIONS_REQUEST_READ_EXTERNAL_STORAGE?
Індивідуальний

17

Ви повинні попросити дозволу під час виконання:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
        && ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
    ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
            REQUEST_PERMISSION);
    dialog.dismiss();
    return;
}

А у зворотному дзвінку нижче ви можете отримати доступ до сховища без проблем.

@Override
public void onRequestPermissionsResult(final int requestCode, @NonNull final String[] permissions, @NonNull final int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if (requestCode == REQUEST_PERMISSION) {
        if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            // Permission granted.
        } else {
            // User refused to grant permission.
        }
    }
}

7

Крок 1: додайте дозвіл на android manifest.xml

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

Крок 2: метод onCreate ()

int permissionCheck = ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE);

    if (permissionCheck != PackageManager.PERMISSION_GRANTED) {
        ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, MY_PERMISSIONS_REQUEST_READ_MEDIA);
    } else {
        readDataExternal();
    }

Крок 3: перевизначити метод onRequestPermissionsResult, щоб отримати зворотний виклик

 @Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {
        case MY_PERMISSIONS_REQUEST_READ_MEDIA:
            if ((grantResults.length > 0) && (grantResults[0] == PackageManager.PERMISSION_GRANTED)) {
                readDataExternal();
            }
            break;

        default:
            break;
    }
}

Примітка: readDataExternal () - це метод отримання даних із зовнішнього сховища.

Дякую.


1
Думаю, ви втратили тут потрібний дозвіл: READ_EXTERNAL_STORAGE.
Renascienza,

4

У мене також був подібний журнал помилок, і ось що я зробив -

  1. У методі onCreate ми запитуємо діалогове вікно для перевірки дозволів

    ActivityCompat.requestPermissions(MainActivity.this,
    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},1);
    
  2. Метод перевірки результату

    @Override
    public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {
        case 1: {
            // If request is cancelled, the result arrays are empty.
            if (grantResults.length > 0
                    && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // permission granted and now can proceed
             mymethod(); //a sample method called
    
            } else {
    
                // permission denied, boo! Disable the
                // functionality that depends on this permission.
                Toast.makeText(MainActivity.this, "Permission denied to read your External storage", Toast.LENGTH_SHORT).show();
            }
            return;
        }
        // add other cases for more permissions
        }
    }
    

Офіційна документація щодо запиту дозволів на виконання


1
я ним користувався, і він працював чудово, чи працює він на всіх андроїд-пристроях?
Mahdi Hraybi

1

Чи вирішено вашу проблему? Який ваш цільовий SDK? Спробуйте додати android;maxSDKVersion="21"до<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />


Працював у мене за допомогою Cordova, встановивши лише максимальну версію SDK в AndroidManifest.xml на 22 (замість 23)
Джордж Каган

0

http://developer.android.com/reference/android/provider/MediaStore.Audio.AudioColumns.html

Спробуйте змінити лінію contentResolver.query

//change it to the columns you need
String[] columns = new String[]{MediaStore.Audio.AudioColumns.DATA}; 
Cursor cursor = contentResolver.query(uri, columns, null, null, null);

публічний статичний фінальний рядок MEDIA_CONTENT_CONTROL

Не для використання сторонніми програмами через конфіденційність споживання медіа

http://developer.android.com/reference/android/Manifest.permission.html#MEDIA_CONTENT_CONTROL

спробувати видалити <uses-permission android:name="android.permission.MEDIA_CONTENT_CONTROL"/>перший і повторити спробу?


Дякую, я спробував вашу пропозицію, але вона зірвалася там же ... Я використовую Mac, чи може це бути проблемою?
Slim Sim

оновлено, спробувати видалити <uses-permission android:name="android.permission.MEDIA_CONTENT_CONTROL"/>і спробувати? Це може створити дивну поведінку, оскільки ваш запит на отримання дозволу не дозволений
Дерек Фунг,

дякую, я спробував видалити цей дозвіл, але він падає в тому ж місці з тією ж помилкою ...
Slim Sim

0

Будь ласка, перевірте нижче код, за допомогою якого Ви можете знайти всі музичні файли з sdcard:

public class MainActivity extends Activity{

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_animations);
    getAllSongsFromSDCARD();

}

public void getAllSongsFromSDCARD() {
    String[] STAR = { "*" };
    Cursor cursor;
    Uri allsongsuri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
    String selection = MediaStore.Audio.Media.IS_MUSIC + " != 0";

    cursor = managedQuery(allsongsuri, STAR, selection, null, null);

    if (cursor != null) {
        if (cursor.moveToFirst()) {
            do {
                String song_name = cursor
                        .getString(cursor
                                .getColumnIndex(MediaStore.Audio.Media.DISPLAY_NAME));
                int song_id = cursor.getInt(cursor
                        .getColumnIndex(MediaStore.Audio.Media._ID));

                String fullpath = cursor.getString(cursor
                        .getColumnIndex(MediaStore.Audio.Media.DATA));

                String album_name = cursor.getString(cursor
                        .getColumnIndex(MediaStore.Audio.Media.ALBUM));
                int album_id = cursor.getInt(cursor
                        .getColumnIndex(MediaStore.Audio.Media.ALBUM_ID));

                String artist_name = cursor.getString(cursor
                        .getColumnIndex(MediaStore.Audio.Media.ARTIST));
                int artist_id = cursor.getInt(cursor
                        .getColumnIndex(MediaStore.Audio.Media.ARTIST_ID));
                System.out.println("sonng name"+fullpath);
            } while (cursor.moveToNext());

        }
        cursor.close();
    }
}


}

Я також додав наступний рядок у файл AndroidManifest.xml, як показано нижче:

<uses-sdk
    android:minSdkVersion="16"
    android:targetSdkVersion="17" />

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

Дякую, я пробував щось подібне раніше, але я спробував ваш код, і він аварійно завершує роботу на "cursor = managedQuery (allsongsuri, STAR, selection, null, null);", така ж помилка, як і раніше. До речі; managedQuery застаріла, але вона по- , як і раніше падає , якщо я змінити його на «contentResolver.query», які мають ті ж вхідні параметри ...
Тонкий Sim

Привіт, я можу отримати всі пісні sdcadr, використовуючи наведений вище код. з пристроєм Android версії 4.2.2, а також з версією 5.0.2. У якому Ви намагаєтесь. Яку помилку ви отримали, опишіть у версії Android
Раджан Бхавсар

Я оновив запитання стеком-трасуванням, сподіваюся, що це допоможе. Що ви маєте на увазі під версією Android? (Я використовую apk 23)
Slim Sim

Добре, дозвольте мені надати повний вихідний код і перевірити це ще раз.
Раджан Бхавсар

якщо ви хочете протестувати для себе, завантажте zip сюди: drive.google.com/open?id=0B2AV8VRk9HUeVzFXSWVRZU9TRUk, коли ви його запускаєте, натисніть ліву кнопку ініціалізації, щоб створити сканування.
Slim Sim

0

На додаток до всіх відповідей. Ви також можете вказати minsdk, який застосовуватиметься з цією анотацією

@TargetApi(_apiLevel_)

Я використав це для того, щоб прийняти цей запит, навіть мій minsdk має 18. Що він робить, це те, що метод працює лише тоді, коли пристрій націлений на "_apilevel_" і верхній. Ось мій метод:

    @TargetApi(23)
void solicitarPermisos(){

    if (ContextCompat.checkSelfPermission(this,permiso)
            != PackageManager.PERMISSION_GRANTED) {

        // Should we show an explanation?
        if (shouldShowRequestPermissionRationale(
                Manifest.permission.READ_EXTERNAL_STORAGE)) {
            // Explain to the user why we need to read the contacts
        }

        requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
                1);

        // MY_PERMISSIONS_REQUEST_READ_EXTERNAL_STORAGE is an
        // app-defined int constant that should be quite unique

        return;
    }

}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.