[iOS swift应用滚动到底部时终止在TableView上

问题描述 投票:0回答:2

我正在使用Firebase在我的iOS应用中填充TableView。前几个对象已加载,但是一旦到达列表中的第三个项目,应用程序就会崩溃,但例外:

'NSRangeException', reason: '*** __boundsFail: index 3 beyond bounds [0 .. 2]'

我知道这意味着我要在不包含索引的索引处引用数组,但是我不知道为什么。

我用TableView创建TableViewController并像这样初始化它:

override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        print(posts.count)
        return posts.count
    }


    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let post = posts[indexPath.row]
        print(post)
        let cell = tableView.dequeueReusableCell(withIdentifier: K.cellIdentifier, for: indexPath) as! PostCell

        let firstReference = storageRef.child(post.firstImageUrl)
        let secondReference = storageRef.child(post.secondImageUrl)

        cell.firstTitle.setTitle(post.firstTitle, for: .normal)
        cell.secondTitle.setTitle(post.secondTitle, for: .normal)
        cell.firstImageView.sd_setImage(with: firstReference)
        cell.secondImageView.sd_setImage(with: secondReference)

        // Configure the cell...

        return cell
    }

[我相信第一个函数创建一个数组,数组中包含posts中的对象数,第二个函数将值分配给单元格的模板。第一种方法中的print语句将打印4,这是从firebase检索的正确对象数。我认为这意味着将创建一个数组,其中包含4个要显示在TableView中的对象。这是真正令人困惑的地方,因为错误指出数组中只有3个对象。我是否误解了TableView的实例化?

这里是填充TableView的代码:

func loadMessages(){
        db.collectionGroup("userPosts")
            .addSnapshotListener { (querySnapshot, error) in

            self.posts = []

            if let e = error{
                print("An error occured trying to get documents. \(e)")
            }else{
                if let snapshotDocuments = querySnapshot?.documents{
                    for doc in snapshotDocuments{
                        let data = doc.data()
                        if let firstImage = data[K.FStore.firstImageField] as? String,
                            let firstTitle = data[K.FStore.firstTitleField] as? String,
                            let secondImage = data[K.FStore.secondImageField] as? String,
                            let secondTitle = data[K.FStore.secondTitleField] as? String{
                            let post = Post(firstImageUrl: firstImage, secondImageUrl: secondImage, firstTitle: firstTitle, secondTitle: secondTitle)
                            self.posts.insert(post, at: 0)
                            print("Posts: ")
                            print(self.posts.capacity)
                            DispatchQueue.main.async {
                                self.tableView.reloadData()
                            }
                        }
                    }
                }
            }
        }

该应用会生成并运行并显示前几项,但是当我滚动到列表底部时会崩溃。非常感谢您的帮助。

编辑:

override func viewDidLoad() {
        super.viewDidLoad()
        tableView.dataSource = self
        tableView.register(UINib(nibName: K.cellNibName, bundle: nil), forCellReuseIdentifier: K.cellIdentifier)
        loadMessages()
    }
ios swift firebase tableview
2个回答
1
投票

您收到越界错误,因为您正在危险地填充数据源。您必须记住,表视图在滚动时会不断添加和删除单元格,这使得更新其数据源成为一项敏感任务。您在每次文档迭代时重新加载表,并在数据源的索引0处插入一个新元素。更新期间的任何滚动都将引发界外错误。

因此,请填充一个临时数据源,并在就绪后将其移交给实际数据源(然后立即重新加载表,在更改后的数据源与从该数据源获取的活动滚动之间不留任何空间。

private var posts = [Post]()
private let q = DispatchQueue(label: "userPosts") // serial queue

private func loadMessages() {
    db.collectionGroup("userPosts").addSnapshotListener { [weak self] (snapshot, error) in
        self?.q.async { // go into the background (and in serial)
            guard let snapshot = snapshot else {
                if let error = error {
                    print(error)
                }
                return
            }
            var postsTemp = [Post]() // setup temp collection
            for doc in snapshot.documents {
                if let firstImage = doc.get(K.FStore.firstImageField) as? String,
                    let firstTitle = doc.get(K.FStore.firstTitleField) as? String,
                    let secondImage = doc.get(K.FStore.secondImageField) as? String,
                    let secondTitle = doc.get(K.FStore.secondTitleField) as? String {
                    let post = Post(firstImageUrl: firstImage, secondImageUrl: secondImage, firstTitle: firstTitle, secondTitle: secondTitle)
                    postsTemp.insert(post, at: 0) // populate temp
                }
            }
            DispatchQueue.main.async { // hop back onto the main queue
                self?.posts = postsTemp // hand temp off (replace or append)
                self?.tableView.reloadData() // reload
            }
        }
    }
}

此外,我将在后台处理此问题(Firestore在主队列上返回),并且仅在修改数据源后才重新加载表。


0
投票

经过一些摆弄并实现@bsod的响应后,我得以使我的项目开始运行。解决方案是在Main.Storyboard中的属性检查器下,我必须将内容设置为动态原型。

© www.soinside.com 2019 - 2024. All rights reserved.