从 MPMediaLibrary 中获取大量歌曲到 UITableView

Fetching large number of songs from MPMediaLibrary into a UITableView

我有一个 table 视图,其中包含音乐库中所有歌曲的列表。 起初我正在获取所有歌曲并将它们的信息保存到这样的数组中:

var songs = [Song]()
private func loadSongs(){
    let itunesSongs = MPMediaQuery.songs().items
    if itunesSongs == nil {
        return;
    }
    for song in itunesSongs!{
        let artist = song.albumArtist
        let trackTitle = song.title
        let image = song.artwork?.image(at: CGSize(width: 90, height: 90))
        let url = song.assetURL?.absoluteString ?? ""
        let song1 = Song(title: trackTitle ?? "", artist: artist ?? "", image: image, url: url)
        songs.append(song1)
    }
}

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cellIdentifier = "SongTableViewCell"
        guard let cell = tableView.dequeueReusableCell(withIdentifier: cellIdentifier, for: indexPath) as? SongCell  else {
            fatalError("The dequeued cell is not an instance of SongCell.")
        }
        let song = playerManager.songs[indexPath.row]
        cell.setAttributes(song: song)

        cell.preservesSuperviewLayoutMargins = false
        return cell
    }

当我在拥有超过 10000 首歌曲的设备上测试时,启动应用程序大约需要 5-10 秒。所以我修改了填充 table 视图的方式,如下所示:

var itunesSongs: MPMediaQuery.songs().items
func getSong(index: Int) -> Song {
    if(index >= songs.count){
        let song = itunesSongs![index]
        let ans = Song(title: song.title ?? "", artist: song.albumArtist ?? "", image: song.artwork?.image(at: CGSize(width: 90, height: 90)), url: song.assetURL?.absoluteString ?? "")
        songs.append(ans)
    }
    return songs[index]
}

所以我会在 tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) 中使用 getSong() 而不是使用 songs 数组。

问题已解决,应用正常启动。但是,偶尔,我会在 return songs[index] 行上得到 Fatal error: Index out of range。 当我快速滚动时会发生这种情况,并且每次都会发生不同的索引。我试图一次获取 100 首歌曲而不是 1 首,但这也没有解决问题。

我正在考虑使用后台线程来填充 songs 数组,但不确定这是否是正确的方法。

问题是 table 视图没有按顺序调用 getSong()。例如,table 视图可以调用 getSong(6) 然后 getSong(3)。我将函数更新为:

func getSong(index: Int) -> Song {
    while (index >= songs.count){
        let song = itunesSongs![songs.count]

        let ans = Song(title: song.title ?? "", artist: song.albumArtist ?? "", image: song.artwork?.image(at: CGSize(width: 90, height: 90)), url: song.assetURL?.absoluteString ?? "")
        songs.append(ans)
    }
    return songs[index]
}

一种可能的解决方案

根本不要使用歌曲数组。也许将大量项目 (10000) 添加到歌曲数组中需要时间。相反

像这样获取viewdidLoad()中的歌曲

let itunesSongs = MPMediaQuery.songs().items

在 cellforRowAt 中,从 itunesSongs 获取歌曲数据(标题、艺术家等)填充您的单元格并 return。像这样的东西

 override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

    let cellIdentifier = "SongTableViewCell"
    guard let cell = tableView.dequeueReusableCell(withIdentifier: cellIdentifier, for: indexPath) as? SongCell  else {
        fatalError("The dequeued cell is not an instance of SongCell.")
    }
    let song = itunesSongs[indexPath.row]
    cell.setAttributes(song: song)

    cell.preservesSuperviewLayoutMargins = false
    return cell
}