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

问题在于,当我添加Genre列[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$1.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和[不推荐使用的GENRE_KEY常量。我可以使用MediaMetadataRetriever获取体裁值,但是比使用游标要花更长的时间。

在这里做错什么了吗?

解决方法

我找到了解决我问题的方法,您可以先使用以下方法声明流派游标:

genreCursor = resolver.query(
    Genres.EXTERNAL_CONTENT_URI,arrayOf(
        Genres._ID,// 0
        Genres.NAME // 1
    ),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",MediaStore.Audio.Media.DEFAULT_SORT_ORDER
)

针对每种流派重复此操作,并且可以以奇怪的方式向流派加载音乐。