StackOverflow 的 friend 们好。
我的应用程序上有一个聊天屏幕,它根据数组的实际大小执行插入和删除。看这个:
func addObject(object: Object?) {
if comments == nil || object == nil || object?.something == nil || object?.anything == nil {
return
}
self.objectsTableView.beginUpdates()
if self.objects!.count == 10 {
self.objects?.removeAtIndex(9)
self.objectsTableView.deleteRowsAtIndexPaths([NSIndexPath(forRow : 9, inSection: 0)], withRowAnimation: .Right)
}
self.objects?.insert(object!, atIndex: 0)
self.objectsTableView.insertRowsAtIndexPaths([NSIndexPath(forRow : 0, inSection: 0)], withRowAnimation: .Right)
self.objectsTableView.endUpdates()
}
但是经过一些压力测试后,日志通知:
Invalid update: invalid number of rows in section 0. The number of rows contained in an existing section after the update (1) must be equal to the number of rows contained in that section before the update (10), plus or minus the number of rows inserted or deleted from that section (1 inserted, 0 deleted) and plus or minus the number of rows moved into or out of that section (0 moved in, 0 moved out).
我不知道发生了什么,只有在插入对象非常极端时才会发生这种情况,例如每 0.2 秒插入一个。
有人知道我能做到吗?
模型不匹配
The number of rows contained in an existing section after the update (1) must be equal to the number of rows contained in that section before the update (10), plus or minus the number of rows inserted or deleted from that section (1 inserted, 0 deleted)
对于通情达理的人来说,UITableView
认为你应该有 11 行:
10
更新前 + 1
插入。
number of rows contained in an existing section after the update (1)
...指的是 numberOfRowsInSection
正在为节 0
返回 1
,这表明 objects
数组不同步,假设您使用以下内容:
override func tableView(tableView: UITableView,
numberOfRowsInSection section: Int) -> Int {
return objects.count
}
使用 NSFetchedResultsController
一个干净的解决方案是使用 NSFetchedResultsController
作为模型和 UI 之间的接口(interface)。它对样板代码进行了深入研究,是确保线程安全的绝佳平台。 Documentation here .
注意:
效果不错!单元格似乎在旋转到顶部。
我无法使用 Gist 打破它您制作,也没有安排多个并发测试。您的 Object
数组必须有恶意访问。
演示
这个简化版本有效。只需将 doPlusAction
Hook 到按钮操作并观看它循环:
class TableViewController: UITableViewController {
var objects:[Int] = [0,1,2,3,4]
var insertions = 5
@IBAction func doPlusAction(sender: AnyObject) {
tableView.beginUpdates()
objects.removeAtIndex(4)
tableView.deleteRowsAtIndexPaths([NSIndexPath(forRow: 4, inSection: 0)], withRowAnimation: .Right)
objects.insert(insertions++, atIndex: 0)
tableView.insertRowsAtIndexPaths([NSIndexPath(forRow: 0, inSection: 0)], withRowAnimation: .Right)
tableView.endUpdates()
let delay = 0.1 * Double(NSEC_PER_SEC) //happens the same with this too, when reach 100-150 items
let time = dispatch_time(DISPATCH_TIME_NOW, Int64(delay))
dispatch_after(time, dispatch_get_main_queue()) { () -> Void in
self.doPlusAction(self)
}
}
override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return objects.count
}
override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("reuseIdentifier", forIndexPath: indexPath)
cell.textLabel!.text = "Cell \(objects[indexPath.row])"
return cell
}
}
关于ios - TableView 上的无效更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35441661/
欢迎光临 OGeek|极客世界-中国程序员成长平台 (https://ogeek.cn/) | Powered by Discuz! X3.4 |