Próbuję uzyskać dostęp do plików multimedialnych (muzyki) na urządzeniu użytkownika, aby je odtworzyć; łatwa aplikacja do odtwarzania muzyki „Hello world”.
Skorzystałem z kilku samouczków i zasadniczo podają ten sam kod. Ale to nie zadziała; ciągle się zawiesza i mówi mi:
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()
....
Oto mój plik manifestu:
<?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>
To jest moja metoda 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());
}
}
Próbowałem:
Aby umieścić to w różnych miejscach w pliku manifestu:
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE”/>
Aby dodać android: maxSdkVersion w warunkach odczytu zewnętrznej pamięci masowej:
<uses-permission
android:name="android.permission.READ_EXTERNAL_STORAGE"
android:maxSdkVersion="21" />
Aby umieścić to w manifeście / aplikacji / tagu aktywności:
android:exported=“true”
Aby umieścić grantUriPremission między uri i cursro w javamethod:
grantUriPermission(null, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION);
Aby tego użyć, nie ulegnie awarii, ale kursor stanie się pusty:
uri = MediaStore.Audio.Media.getContentUri("EXTERNAL_CONTENT_URI”);
Aby użyć WEWNĘTRZNEGO identyfikatora uri zawartości, działa to zgodnie z oczekiwaniami, ale daje tylko „dźwięki systemu operacyjnego”, takie jak dźwięk migawki, dźwięk niskiego poziomu baterii, kliknięcie przycisku itp .:
uri = MediaStore.Audio.Media.INTERNAL_CONTENT_URI;
Proszę o pomoc, to nie powinien być trudny problem, który znam, ale czuję się jak początkujący!
Przeczytałem i próbowałem (lub uważałem, że nie mają zastosowania do mojego problemu):
- Uprawnienie Android READ_EXTERNAL_STORAGE nie działa
- Wymagaj pozwolenia tylko dla starszych wersji Androida: maxSdkVersion nie działa?
- Uzyskaj nazwę pliku i ścieżkę z URI z mediastore
- Android KitKat securityException podczas próby odczytu z MediaStore
- Android: java.lang.SecurityException: Odmowa uprawnień: uruchom zamiar
Ślad stosu:
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
Odpowiedzi:
Masz dwa rozwiązania swojego problemu. Szybkim jest obniżenie targetApi do 22 (plik build.gradle). Po drugie, użyj nowego i wspaniałego modelu prośby o pozwolenie:
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 znaleziony tutaj: https://developer.android.com/training/permissions/requesting.html
Rozwiązanie 2: Jeśli to nie działa, spróbuj tego:
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;
}
a następnie w oddzwonieniu
@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. } } }
to jest z komentarzy. dzięki
źródło
READ_EXTERNAL_STORAGE
. Jak w tej odpowiedzi .Musisz poprosić o pozwolenie w czasie wykonywania:
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; }
A w wywołaniu zwrotnym poniżej możesz uzyskać dostęp do pamięci bez problemu.
@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. } } }
źródło
Krok 1: Dodaj uprawnienia do pliku manifest.xml w systemie Android
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
Krok 2: metoda 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(); }
Krok 3: Zastąp metodę onRequestPermissionsResult, aby uzyskać wywołanie zwrotne
@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; } }
Uwaga: metoda readDataExternal () służy do pobierania danych z pamięci zewnętrznej.
Dzięki.
źródło
Miałem też podobny dziennik błędów i oto co zrobiłem-
W metodzie onCreate żądamy okna dialogowego do sprawdzenia uprawnień
ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},1);
Metoda sprawdzenia wyniku
@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 } }
Oficjalna dokumentacja dotycząca żądania uprawnień środowiska wykonawczego
źródło
Czy Twój problem został rozwiązany? Jaki jest Twój docelowy SDK? Spróbuj dodać
android;maxSDKVersion="21"
do<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
źródło
http://developer.android.com/reference/android/provider/MediaStore.Audio.AudioColumns.html
Spróbuj zmienić linię
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);
http://developer.android.com/reference/android/Manifest.permission.html#MEDIA_CONTENT_CONTROL
spróbować usunąć
<uses-permission android:name="android.permission.MEDIA_CONTENT_CONTROL"/>
pierwszą i spróbować ponownie?źródło
<uses-permission android:name="android.permission.MEDIA_CONTENT_CONTROL"/>
i spróbować? Może to powodować dziwne zachowanie, ponieważ Twoja aplikacja nie prosi o pozwolenieSprawdź poniższy kod, dzięki któremu możesz znaleźć wszystkie pliki muzyczne z 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(); } } }
Dodałem również następujący wiersz w pliku AndroidManifest.xml, jak poniżej:
<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" />
źródło
Oprócz wszystkich odpowiedzi. Możesz również określić minsdk do zastosowania z tą adnotacją
@TargetApi(_apiLevel_)
Użyłem tego, aby zaakceptować tę prośbę, nawet gdy mój minsdk ma 18 lat. To, co robi, to fakt, że metoda działa tylko wtedy, gdy urządzenie jest przeznaczone na „_apilevel_” i wyższe. Oto moja metoda:
@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; } }
źródło