当拉到刷新时,我发现这个奇怪的应用程序崩溃。
我的代码如下:
var posts: [Posts] = []
override func viewDidLoad() {
super.viewDidLoad()
// refreshControl -> pull to refresh handler
let refreshControl = UIRefreshControl()
refreshControl.addTarget(self,
action: #selector(Main_TVC.getData),
for: UIControlEvents.valueChanged)
self.refreshControl = refreshControl
getData()
}
override func numberOfSections(in tableView: UITableView) -> Int {
return 1
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return posts.count
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "cell",
for: indexPath) as! PostsTableViewCell
cell.titleLabel.text = posts[indexPath.row].postTitle
cell.bodyLabel.text = posts[indexPath.row].postBody
return cell
}
func getData() {
self.posts.removeAll()
// queries the backend and fills the data - sorry for code omission
refresh()
}
func refresh() {
self.tableView.reloadData()
self.refreshControl?.endRefreshing()
}
该应用程序运行正常,即使我下拉刷新,一切都运行完美,但如果长时间下拉刷新,如拉下几乎击中屏幕的底部,应用程序崩溃并提示以下错误:
致命错误:索引超出范围
就行了
cell.titleLabel.text = posts [indexPath.row] .postTitle
如果我按如下方式打印帖子计数和indexPath:
打印(" posts.count =(posts.count)") 打印(" indexPath.row =(indexPath.row)")
当我以正常方式拉下来时,它会打印出正确的数据,但是如果我拉下来就像一个长时间的拉动,如果它在崩溃时提示它,则通过整个屏幕
posts.count = 0
indexPath.row = 2
这种事情从未发生在我使用refreshControl的地方,就像我在这里使用的那样。
希望我的信息是可以理解的,主要是长期刷新问题。
答案 0 :(得分:8)
你的问题是你在getData
中做的第一件事就是删除self.posts
中的所有帖子,但你没有(大概是因为缺少代码)重新加载表格视图,所以现在数组中的帖子数(0)以及tableview *认为在数组中的帖子数(不为零)是不同的,因此您会遇到数组边界崩溃。
在reloadData
之后调用self.posts.removeAll()
可以解决问题,但会导致表格查看'闪烁'因为它刷新了空,然后用新数据重新绘制。
由于您还没有显示getData
的完整代码,因此我无法提供您需要的确切代码,但它应该是这样的:
func getData() {
fetchPostsWithCompletion() {
if let tempPosts = dataFromNetwork {
self.posts = tempPosts
} else {
self.posts.removeAll()
}
self.refresh() // Dispatch this on the main queue if your completion handler is not already on the main queue
}
}
这样,在获得新数据之前,您不会操纵后备阵列。
答案 1 :(得分:0)
我遇到了同样的问题。我的发现是,延迟removeAll()
数组上posts
的执行,可以使表格视图在其count
上更新。
func getData() {
self.delayExecutionByMilliseconds(500) {
self.posts.removeAll()
}
// queries the backend and fills the data - sorry for code omission
refresh()
}
fileprivate func delayExecutionByMilliseconds(_ delay: Int, for anonFunc: @escaping () -> Void) {
let when = DispatchTime.now() + .milliseconds(delay)
DispatchQueue.main.asyncAfter(deadline: when, execute: anonFunc)
}