安卓的READ_EXTERNAL_STORAGE权限

97

我试图访问用户设备上的媒体文件(音乐),以播放它们; 一个简单的“hello world”-音乐播放器应用程序。

我已经按照一些教程,它们基本上都给出相同的代码。但是它不起作用; 它会崩溃并告诉我:

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" />
将此放入清单文件 / 应用程序 / 活动标签中:
android:exported=“true

在Java方法中,在uri和cursor之间添加grantUriPermission:

grantUriPermission(null, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION);

使用此功能不会导致崩溃,但光标会变成 null。

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

很奇怪,当您在清单中没有给出以下行 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> 时,会显示错误。您是否尝试读取受其他应用程序保护的某些文件? - Sunil Sunny
你尝试过清理项目吗? - Sunil Sunny
@sunilsunny 我并没有试图读取受保护的文件,至少我不知道有这样的文件存在,只是一个简单的媒体播放器。是的,我已经尝试过清理它,重启电脑,生成签名APK并将其发布到Google Play并作为测试人员访问,但都没有成功... - Slim Sim
@SharpEdge;我的AVD是标准的Nexus 5,API 23。我的模块gradle文件中有以下内容: compileSdkVersion 23 buildToolsVersion "23.0.0" minSdkVersion 14 targetSdkVersion 23 因此我会说是23。 - Slim Sim
@k3b 我的AVD上没有其他音频,但我希望我的应用程序在没有崩溃的情况下显示nrSongs=0。在我的移动设备上,我有歌曲,可以使用Google Play Music和从Play商店下载的另一个音乐播放器打开。所以,是的-我可以。“文件是否受DRM保护?”-我不知道,我只想访问所有我被允许访问的音乐文件...就像一个音乐播放器...“它是否位于受保护的目录中...”与之前相同的答案。注意:我不将音频包含为应用程序的一部分,我想从用户的媒体库中播放音频。 - Slim Sim
显示剩余2条评论
10个回答

98

你有两种解决方案。快速的方法是将targetApi降低到22(build.gradle文件中)。第二种方法是使用新的,神奇的ask-for-permission模型:

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
当然可以!谷歌最新的Android引入了新的权限模型!非常感谢,我会尽快测试并在有效时标记为正确!(我开始觉得这被遗忘了!) - Slim Sim
1
所请求的权限应该是“READ_EXTERNAL_STORAGE”。就像这个答案中所示。 - maclir
1
@Renascienza,是的,你说得对。有人以错误的方式编辑了这个答案。 - piotrpo
checkSelfPermissions()这个方法是从哪里来的?我在这里遇到了红线错误。 - TheQ
2
MY_PERMISSIONS_REQUEST_READ_EXTERNAL_STORAGE 是什么? - Solo
显示剩余2条评论

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

9
步骤1:在 Android manifest.xml 中添加权限。
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

第二步: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();
    }

第三步:重写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

6

从Android 13开始,您需要使用READ_MEDIA_IMAGES

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

所以你需要在API级别低于33的情况下以编程方式请求READ_EXTERNAL_STORAGE权限,并在API级别等于或高于33的情况下请求READ_MEDIA_IMAGES权限,这里是使用新API的示例。
binding.btnchangeProfilePicture.setOnClickListener {
                val readImagePermission =
                    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU)
                        Manifest.permission.READ_MEDIA_IMAGES else Manifest.permission.READ_EXTERNAL_STORAGE
                requestPermission.launch(readImagePermission)
            }
    
private val requestPermission = registerForActivityResult(ActivityResultContracts.RequestPermission()) { isGranted ->
                    if (isGranted) {
                       
                    } else {
                        
                    }
                }

1
这个有效。谢谢。这应该被接受的答案。 - Shurvir Mori
1
只需添加 <uses-permission android:name="android.permission.READ_MEDIA_IMAGES" /> 对我来说就可以。 - Shurvir Mori

5
我也遇到了类似的错误日志,这是我所做的 -
  1. In onCreate method we request a Dialog Box for checking permissions

    ActivityCompat.requestPermissions(MainActivity.this,
    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},1);
    
  2. Method to check for the result

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

1
你的问题解决了吗?你的目标SDK是什么?尝试在<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />中添加android;maxSDKVersion="21"

通过在 AndroidManifest.xml 中仅将最大 SDK 版本设置为 22(而不是 23),我使用 Cordova 工作。 - George Kagan

1

在2021年,我面临这个问题,所以我将使用清单文件解决这个问题。我会在应用程序中添加一个标签:

<application 
android:requestLegacyExternalStorage="true"
/>

注意:此方法仅适用于Android 10及以下版本,更多信息请参见此处


requestLegacyExternalStorage - Ateş Danış

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

}

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

public static final String 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"/> 并重试。这可能会导致奇怪的行为,因为您的应用程序请求了一个不允许的权限。 - Derek Fung
谢谢,我尝试去掉那个权限,但它在同一位置以相同的错误崩溃... - Slim Sim

0
请查看下面的代码,使用它可以从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();
    }
}


}

我还在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”,它仍然会崩溃,因为它具有相同的输入参数... - Slim Sim
嗨,我能够使用上述代码获取所有SD卡中的歌曲。在Android 4.2.2设备版本和5.0.2版本上都可以实现。请描述您在尝试中遇到的错误,并说明您所使用的Android版本。 - Rajan Bhavsar
我已更新了问题的堆栈跟踪,希望有所帮助。您说的 Android 版本是什么意思?(我使用的是 apk 23) - Slim Sim
好的,让我给你提供完整源代码并再次检查它。 - Rajan Bhavsar
如果您想自行测试,请下载此处的压缩文件: https://drive.google.com/open?id=0B2AV8VRk9HUeVzFXSWVRZU9TRUk 运行后,按下左侧的初始化按钮即可触发崩溃。 - Slim Sim

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接