UITableView 意外反弹 beginUpdates()/endUpdates()/performBatchUpdates()
Posted
技术标签:
【中文标题】UITableView 意外反弹 beginUpdates()/endUpdates()/performBatchUpdates()【英文标题】:UITableView unexpectedly bounces with beginUpdates()/endUpdates()/performBatchUpdates() 【发布时间】:2018-02-07 07:32:18 【问题描述】:我有一个非常直截了当的 UITableViewController
/NSFetchedResultsController
案例。它来自 Xcode Master-Detail App 示例代码,很容易重现。
我有一个带有 1 个实体和 1 个字符串属性的 CoreData 模型。它显示在UITableViewController
中。我使用.subtitle
系统单元格类型。
通过选择一行,我只需更新字符串属性。
所以我的问题是,当我为表格视图添加足够的行以滚动(在带有导航栏的 iPhone 5s 上为 10-11 行),然后我向下滚动并选择任何行时,表格视图会上下弹跳。
如果行数较少(少于 10 行)或行数较多(12 行或更多),则行为正常。
因此,问题似乎在滚动视图的极限。
如果我不使用beginUpdates()
/endUpdates()
,问题就消失了,但我失去了它们的优势。
这是what happens的视频链接
class TableViewController: UITableViewController, NSFetchedResultsControllerDelegate
var managedObjectContext: NSManagedObjectContext? = nil
@objc func insertNewObject(_ sender: Any)
let context = self.fetchedResultsController.managedObjectContext
let newEvent = Event(context: context)
newEvent.aString = "a String"
try? context.save()
override func viewDidLoad()
super.viewDidLoad()
let addButton = UIBarButtonItem(barButtonSystemItem: .add, target: self, action: #selector(insertNewObject(_:)))
navigationItem.rightBarButtonItem = addButton
override func numberOfSections(in tableView: UITableView) -> Int
return fetchedResultsController.sections?.count ?? 0
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int
let sectionInfo = fetchedResultsController.sections![section]
return sectionInfo.numberOfObjects
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath)
let event = fetchedResultsController.object(at: indexPath)
configureCell(cell, withEvent: event)
return cell
func configureCell(_ cell: UITableViewCell, withEvent event: Event)
cell.textLabel?.text = event.aString
cell.detailTextLabel?.text = event.aString
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath)
tableView.deselectRow(at: indexPath, animated: true)
let event: Event = self.fetchedResultsController.object(at: indexPath)
event.aString = event.aString! + ""
override func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool
return true
override func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath)
if editingStyle == .delete
let context = fetchedResultsController.managedObjectContext
context.delete(fetchedResultsController.object(at: indexPath))
try? context.save()
var fetchedResultsController: NSFetchedResultsController<Event>
if _fetchedResultsController != nil
return _fetchedResultsController!
let fetchRequest: NSFetchRequest<Event> = Event.fetchRequest()
fetchRequest.fetchBatchSize = 20
let sortDescriptor = NSSortDescriptor(keyPath: \Event.aString, ascending: false)
fetchRequest.sortDescriptors = [sortDescriptor]
let aFetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: self.managedObjectContext!, sectionNameKeyPath: nil, cacheName: nil)
aFetchedResultsController.delegate = self
_fetchedResultsController = aFetchedResultsController
try? _fetchedResultsController!.performFetch()
return _fetchedResultsController!
var _fetchedResultsController: NSFetchedResultsController<Event>? = nil
func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>)
tableView.beginUpdates()
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange sectionInfo: NSFetchedResultsSectionInfo, atSectionIndex sectionIndex: Int, for type: NSFetchedResultsChangeType)
switch type
case .insert:
tableView.insertSections(IndexSet(integer: sectionIndex), with: .automatic)
case .delete:
tableView.deleteSections(IndexSet(integer: sectionIndex), with: .automatic)
default:
return
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?)
switch type
case .insert:
tableView.insertRows(at: [newIndexPath!], with: .automatic)
case .delete:
tableView.deleteRows(at: [indexPath!], with: .automatic)
case .update:
configureCell(tableView.cellForRow(at: indexPath!)!, withEvent: anObject as! Event)
case .move:
configureCell(tableView.cellForRow(at: indexPath!)!, withEvent: anObject as! Event)
tableView.moveRow(at: indexPath!, to: newIndexPath!)
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>)
tableView.endUpdates()
【问题讨论】:
您是否尝试将估计的像元高度设置为正确值?错误的值通常会导致 UI 跳转 【参考方案1】:我遇到了同样的问题,我发现实现 estimatedHeightFor...
方法为我解决了这个问题。该问题似乎源于在表格中使用自动单元格高度,而不是明确定义的每行高度。
我使用的表格既有行又有节的页眉/页脚,所以我需要定义行和页眉的估计高度,这解决了批量更新期间奇怪的弹跳动画。
请注意,为节标题高度返回 0 将使用表格视图的默认值,可能是UITableViewAutomaticDimension
,因此为估计的高度返回一个正数。
我在 Obj-C 中的代码:
-(CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath
return 44; // anything except 0 or UITableViewAutomaticDimension
-(CGFloat)tableView:(UITableView *)tableView estimatedHeightForHeaderInSection:(NSInteger)section
return 18; // anything except 0 or UITableViewAutomaticDimension
-(CGFloat)tableView:(UITableView *)tableView estimatedHeightForFooterInSection:(NSInteger)section
return 18; // anything except 0 or UITableViewAutomaticDimension
【讨论】:
以上是关于UITableView 意外反弹 beginUpdates()/endUpdates()/performBatchUpdates()的主要内容,如果未能解决你的问题,请参考以下文章
UITableView - BeginUpdates/EndUpdates 动画设置
beginUpdates/endUpdates 后出现意外的表格单元格动画
执行“beginUpdates”“endUpdates”时 UITableView 页脚视图的动画问题
UITableView beginUpdates/endUpdates 回调
UITableView "beginUpdates" 和 UICollectionView "performBatchUpdates" 是不是具有相同的行为?