从 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
}
我有一个 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
}