ユーザーデバイスのメディアファイル(音楽)にアクセスして再生しようとしています。簡単な「ハローワールド」音楽プレーヤーアプリ。
私はいくつかのチュートリアルに従っていますが、基本的に同じコードを提供します。しかし、それは機能しません。それはクラッシュし続け、私に言っています:
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”/>
Read external storage premissionでAndroid:maxSdkVersionを追加するには:
<uses-permission
Android:name="Android.permission.READ_EXTERNAL_STORAGE"
Android:maxSdkVersion="21" />
これをマニフェスト/アプリケーション/アクティビティタグに追加するには:
Android:exported=“true”
Javamethodでuriとcursroの間にgrantUriPremissionを配置するには:
grantUriPermission(null, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION);
これを使用するには、クラッシュしませんが、カーソルはnullになります。
uri = MediaStore.Audio.Media.getContentUri("EXTERNAL_CONTENT_URI”);
内部コンテンツuriを使用する場合、これは期待どおりに機能しますが、シャッター音、低バッテリー音、ボタンクリックなどの「OS音」のみを提供します。
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
問題には2つの解決策があります。簡単な方法は、targetApiを22(build.gradleファイル)に下げることです。 2つ目は、新しいすばらしい許可要求モデルを使用することです。
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
実行時に許可を求めています:
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.
}
}
}
ステップ1:Android manifest.xmlに許可を追加する
<uses-permission Android:name="Android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission Android:name="Android.permission.WRITE_EXTERNAL_STORAGE"/>
Step2: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()は、外部ストレージからデータを取得するメソッドです。
ありがとう。
私も同様のエラーログを持っていましたが、これが私がしたことです-
OnCreateメソッドでは、権限を確認するためのダイアログボックスをリクエストします
ActivityCompat.requestPermissions(MainActivity.this,
new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},1);
結果を確認する方法
@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
}
}
ランタイムパーミッションのリクエスト の公式ドキュメント
問題は解決しましたか?ターゲットSDKは何ですか? Android;maxSDKVersion="21"
を<uses-permission Android:name="Android.permission.READ_EXTERNAL_STORAGE" />
に追加してみてください
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"/>
を削除してから再試行してください。
すべての答えに加えて。この注釈で適用する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;
}
}
以下を使用して、それを使用して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" />