无法使用 MediaStore.Audio.AudioColumns 从歌曲中提取流派元数据
Cannot extract genre metadata from song using MediaStore.Audio.AudioColumns
我目前正在为 Android 设备开发音乐播放器。
为了从系统中索引音乐文件,我在 Kotlin 中编写了这个函数来获取手机文件系统中所有音乐的游标,以及使用 MediaStore.Audio.AudioColumns
:
的元数据
private fun getMusicCursor(resolver: ContentResolver): Cursor? {
Log.i(this::class.simpleName, "Getting music cursor.")
return resolver.query(
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.GENRE, // 5
AudioColumns.YEAR, // 6
AudioColumns.TRACK, // 7
AudioColumns.DURATION // 8
),
AudioColumns.IS_MUSIC + "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
}
问题是,当我添加流派列 [AudioColumns.GENRE
] 时,代码总是失败,并且如果在 try-catch 语句之外将抛出此异常:
2020-08-21 09:14:26.077 16658-16715/org.oxycblt.auxio E/AndroidRuntime: FATAL EXCEPTION: DefaultDispatcher-worker-1
Process: org.oxycblt.auxio, PID: 16658
java.lang.IllegalArgumentException: Invalid column genre
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:170)
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:140)
at android.content.ContentProviderProxy.query(ContentProviderNative.java:437)
at android.content.ContentResolver.query(ContentResolver.java:962)
at android.content.ContentResolver.query(ContentResolver.java:890)
at android.content.ContentResolver.query(ContentResolver.java:846)
at org.oxycblt.auxio.music.MusicLoader.getMusicCursor(MusicLoader.kt:95)
at org.oxycblt.auxio.music.MusicLoader.findMusic(MusicLoader.kt:39)
at org.oxycblt.auxio.music.MusicLoader.<init>(MusicLoader.kt:29)
at org.oxycblt.auxio.music.MusicRepository.init(MusicRepository.kt:25)
at org.oxycblt.auxio.loading.LoadingViewModel$startMusicRepo.invokeSuspend(LoadingViewModel.kt:41)
at kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
at kotlinx.coroutines.DispatchedTask.run(Dispatched.kt:241)
at kotlinx.coroutines.scheduling.CoroutineScheduler.runSafely(CoroutineScheduler.kt:594)
at kotlinx.coroutines.scheduling.CoroutineScheduler.access$runSafely(CoroutineScheduler.kt:60)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.run(CoroutineScheduler.kt:740)
这也会影响其他与流派相关的值,例如 GENRE_ID
和 [deprecated] GENRE_KEY
常量。我可以使用 MediaMetadataRetriever
来获取流派值,但这比使用 Cursor 花费的时间要长得多。
我是不是做错了什么?
我找到了解决我的问题的方法,您可以先声明一个 Genre 游标,方法如下:
genreCursor = resolver.query(
Genres.EXTERNAL_CONTENT_URI,
arrayOf(
Genres._ID, // 0
Genres.NAME // 1
),
null, null,
Genres.DEFAULT_SORT_ORDER
)
然后使用光标在系统中创建每个流派的列表。
之后,当您想加载音乐时,您可以创建一个根据流派 ID 过滤的歌曲 Cursor,如下所示:
songCursor = resolver.query(
Genres.Members.getContentUri("external", genreId),
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.YEAR, // 5
AudioColumns.TRACK, // 6
AudioColumns.DURATION // 7
),
AudioColumns.IS_MUSIC + "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
对每种流派重复此操作,您就可以很好地加载音乐流派,尽管方式很奇怪。
我目前正在为 Android 设备开发音乐播放器。
为了从系统中索引音乐文件,我在 Kotlin 中编写了这个函数来获取手机文件系统中所有音乐的游标,以及使用 MediaStore.Audio.AudioColumns
:
private fun getMusicCursor(resolver: ContentResolver): Cursor? {
Log.i(this::class.simpleName, "Getting music cursor.")
return resolver.query(
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.GENRE, // 5
AudioColumns.YEAR, // 6
AudioColumns.TRACK, // 7
AudioColumns.DURATION // 8
),
AudioColumns.IS_MUSIC + "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
}
问题是,当我添加流派列 [AudioColumns.GENRE
] 时,代码总是失败,并且如果在 try-catch 语句之外将抛出此异常:
2020-08-21 09:14:26.077 16658-16715/org.oxycblt.auxio E/AndroidRuntime: FATAL EXCEPTION: DefaultDispatcher-worker-1
Process: org.oxycblt.auxio, PID: 16658
java.lang.IllegalArgumentException: Invalid column genre
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:170)
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:140)
at android.content.ContentProviderProxy.query(ContentProviderNative.java:437)
at android.content.ContentResolver.query(ContentResolver.java:962)
at android.content.ContentResolver.query(ContentResolver.java:890)
at android.content.ContentResolver.query(ContentResolver.java:846)
at org.oxycblt.auxio.music.MusicLoader.getMusicCursor(MusicLoader.kt:95)
at org.oxycblt.auxio.music.MusicLoader.findMusic(MusicLoader.kt:39)
at org.oxycblt.auxio.music.MusicLoader.<init>(MusicLoader.kt:29)
at org.oxycblt.auxio.music.MusicRepository.init(MusicRepository.kt:25)
at org.oxycblt.auxio.loading.LoadingViewModel$startMusicRepo.invokeSuspend(LoadingViewModel.kt:41)
at kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
at kotlinx.coroutines.DispatchedTask.run(Dispatched.kt:241)
at kotlinx.coroutines.scheduling.CoroutineScheduler.runSafely(CoroutineScheduler.kt:594)
at kotlinx.coroutines.scheduling.CoroutineScheduler.access$runSafely(CoroutineScheduler.kt:60)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.run(CoroutineScheduler.kt:740)
这也会影响其他与流派相关的值,例如 GENRE_ID
和 [deprecated] GENRE_KEY
常量。我可以使用 MediaMetadataRetriever
来获取流派值,但这比使用 Cursor 花费的时间要长得多。
我是不是做错了什么?
我找到了解决我的问题的方法,您可以先声明一个 Genre 游标,方法如下:
genreCursor = resolver.query(
Genres.EXTERNAL_CONTENT_URI,
arrayOf(
Genres._ID, // 0
Genres.NAME // 1
),
null, null,
Genres.DEFAULT_SORT_ORDER
)
然后使用光标在系统中创建每个流派的列表。
之后,当您想加载音乐时,您可以创建一个根据流派 ID 过滤的歌曲 Cursor,如下所示:
songCursor = resolver.query(
Genres.Members.getContentUri("external", genreId),
arrayOf(
AudioColumns._ID, // 0
AudioColumns.DISPLAY_NAME, // 1
AudioColumns.TITLE, // 2
AudioColumns.ARTIST, // 3
AudioColumns.ALBUM, // 4
AudioColumns.YEAR, // 5
AudioColumns.TRACK, // 6
AudioColumns.DURATION // 7
),
AudioColumns.IS_MUSIC + "=1", null,
MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)
对每种流派重复此操作,您就可以很好地加载音乐流派,尽管方式很奇怪。