Permiso READ_EXTERNAL_STORAGE para Android

Estoy tratando de acceder a los archivos multimedia (música) en el dispositivo de los usuarios para reproducirlos; una sencilla aplicación de “hello world” -music player.

He seguido algunos tutoriales y básicamente dan el mismo código. Pero no funcionará; sigue estrellándose y diciéndome:

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() .... 

Ahora, este es mi archivo manifiesto:

               

Este es mi método de 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()); } } 

Yo he tratado:

Para poner esto en diferentes lugares en el archivo de manifiesto:

  

Para agregar android: maxSdkVersion at Read premission de almacenamiento externo:

  

Para poner esto en el manifiesto / application / activity-tag:

 android:exported=“true” 

Para poner grantUriPremission entre uri y cursro en el método java:

 grantUriPermission(null, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION); 

Para usar esto, no se bloqueará, pero el cursor se volverá nulo:

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

Para usar el contenido URI INTERNO, esto funciona como se espera, pero solo da “sonidos del sistema operativo”, como sonido de obturador, batería baja, sonido, clic y así:

 uri = MediaStore.Audio.Media.INTERNAL_CONTENT_URI; 

Por ayuda, este no debería ser un problema difícil, pero me siento como un principiante.

He leído y probado (o los considero que no son aplicables para mi problema):

  • El permiso de Android READ_EXTERNAL_STORAGE no funciona
  • Requiere permiso solo para versiones anteriores de Android: maxSdkVersion no funciona.
  • Obtener nombre de archivo y ruta desde URI desde mediastore
  • Android KitKat securityException al intentar leer desde MediaStore
  • Android: java.lang.SecurityException: denegación de permiso: intención de inicio

Stack trace:

 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 

Tienes dos soluciones para tu problema. El más rápido es bajar targetApi a 22 (archivo build.gradle). Segundo, utilizar un nuevo y maravilloso modelo de solicitud de permiso:

 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; } 

Sniplet encontrado aquí: https://developer.android.com/training/permissions/requesting.html

Ha solicitado el permiso en tiempo de ejecución:

 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; } 

Y en la callback a continuación, puede acceder al almacenamiento sin problemas.

 @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. } } } 

Paso 1: agrega permiso en android manifest.xml

   

Paso 2: método 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(); } 

Paso 3: anula el método onRequestPermissionsResult para obtener una callback

  @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; } } 

Nota: readDataExternal () es un método para obtener datos del almacenamiento externo.

Gracias.

También tuve un registro de error similar y esto es lo que hice-

  1. En el método onCreate solicitamos un cuadro de diálogo para verificar los permisos

     ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},1); 
  2. Método para verificar el resultado

     @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 } } 

La documentación oficial para solicitar permisos de tiempo de ejecución

¿Tu problema ha sido resuelto? ¿Cuál es tu SDK objective? Prueba agregar android;maxSDKVersion="21" a

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

Intenta cambiar la línea 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); 

public static final Cadena MEDIA_CONTENT_CONTROL

No debe ser utilizado por aplicaciones de terceros debido a la privacidad del consumo de medios

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

intente eliminar primero e intente de nuevo?

Por favor, compruebe el código a continuación que usa eso. Puede encontrar todos los archivos de música desde la tarjeta SD:

 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(); } } } 

También he agregado la siguiente línea en el archivo AndroidManifest.xml de la siguiente manera:

     

Además de todas las respuestas. También puede especificar el minsdk para aplicar con esta anotación

 @TargetApi(_apiLevel_) 

Lo usé para aceptar esta solicitud, incluso mi minsdk es 18. Lo que hace es que el método solo se ejecuta cuando el dispositivo apunta a “_apilevel_” y superior. Este es mi método:

  @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; } } 

Simplemente vaya a su “configuración”, en “dispositivo” seleccione “aplicaciones” y luego desplácese a su MY_APPLICATION y otorgue el permiso de “almacenamiento”.