0

我寫了簡單的計數器應用程序。它使用CoreData來存儲數據。在模型中,我有一個名爲「計數器」的實體,其屬性爲「值」(Int64)和「位置」(Int16)。第二個屬性是tableView中的位置。用戶可以添加,刪除或增加計數器。我也使用NSFetchedResultsController來填充我的tableView。應用程序的作品,但是當我,例如,添加5個計數器,更改它們的值,刪除第二個和第三個值,並添加一個新的,計數器將不會插入tableView的結尾(這裏是視頻https://youtu.be/XXyuEaobd3c)。我的代碼有什麼問題?NSFetchedResultsController添加錯誤indexPath中的項目

import UIKit 
import CoreData 

class TableViewController: UITableViewController, NSFetchedResultsControllerDelegate { 

let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext 

var frc: NSFetchedResultsController<Counter> = { 
    let fetchRequest: NSFetchRequest<Counter> = Counter.fetchRequest() 
    let sortDescriptor = NSSortDescriptor(key: "position", ascending: true) 
    fetchRequest.sortDescriptors = [sortDescriptor] 
    let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext 
    let frc = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: context, sectionNameKeyPath: nil, cacheName: nil) 
    do { 
     try frc.performFetch() 
    } catch { 
     print(error) 
    } 
    return frc 
}() 


@IBAction func addCounter(_ sender: UIBarButtonItem) { 
    let counter = Counter(context: context) 
    counter.value = 0 
    counter.position = Int16(frc.fetchedObjects!.count) 
    do { 
     try context.save() 
    } catch { 
     print(error) 
    } 

} 

@IBAction func longPressed(_ sender: UILongPressGestureRecognizer) { 
    if sender.state == UIGestureRecognizerState.began { 
     let touchPoint = sender.location(in: self.view) 
     if let indexPath = tableView.indexPathForRow(at: touchPoint) { 
      // your code here, get the row for the indexPath or do whatever you want 
      let counter = frc.object(at: indexPath) 
      counter.value = 0 

      do { 
       try context.save() 
      } catch { 
       print(error) 
      } 
     } 
    } 

} 


override func viewDidLoad() { 
    super.viewDidLoad() 

    frc.delegate = self 
} 


// MARK: - Table view data source 

override func numberOfSections(in tableView: UITableView) -> Int { 
    guard let sections = frc.sections?.count else {return 0} 
    return sections 
} 

override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { 
    guard let sections = frc.sections else {return 0} 
    return sections[section].numberOfObjects 
} 


override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 
    let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath) 
    let counter = frc.object(at: indexPath) 
    cell.textLabel!.text! = String(counter.value) 

    return cell 
} 

func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) { 
    tableView.beginUpdates() 
} 

func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?) { 
    switch type { 
    case .insert: 
     guard let indexPath = newIndexPath else {return} 
     tableView.insertRows(at: [indexPath], with: .fade) 
    case .delete: 
     guard let indexPath = indexPath else {return} 
     tableView.deleteRows(at: [indexPath], with: .fade) 
    case .update: 
     guard let indexPath = indexPath else {return} 
     let cell = tableView.cellForRow(at: indexPath) 
     let counter = frc.object(at: indexPath) 
     cell?.textLabel?.text = String(counter.value) 
    default: 
     break 
    } 
} 

func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) { 
    tableView.endUpdates() 
} 

override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { 
    let counter = frc.object(at: indexPath) 
    counter.value += Int64(1) 

    do {try context.save()} 
    catch {print(error)} 

    tableView.deselectRow(at: indexPath, animated: true) 
} 

    // Override to support editing the table view. 
override func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) { 
    if editingStyle == .delete { 
      let counter = frc.object(at: indexPath) 
      context.delete(counter) 

     do { 
      try context.save() 
     } catch { 
      print(error) 
     } 
    } 
} 

}

回答

-1

你加入在錯誤的方式的位置。 我們假設你有

A1,A2,A3,A4,A5

然後將其刪除A2,A3之後

您有 A1,A4,A5

因此增加新的一個是B3,因爲現在計數爲3 A1,B3,A4,A5

添加另一行成爲B4,因此它可以在A4之前或之後的任何位置。

此訂單歸因於排序上的位置。

你能解決這個問題,通過獲得上次讀取對象,並增加了位置,結果,那麼將是:

A1,A4,A5,B6,B7

相關問題