无法使用 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
)

对每种流派重复此操作,您就可以很好地加载音乐流派,尽管方式很奇怪。