【问题标题】:NSFetchedResultsController - no delegate functions are calledNSFetchedResultsController - 不调用委托函数
【发布时间】:2020-04-29 21:41:20
【问题描述】:

我一直在 Stack 上寻找答案,但我似乎找不到任何有效的方法。问题很简单——我有一个视图控制器,它有一个子表视图控制器。

class ViewController: UIView Controller { 
    var child: UIViewController!
    var managedObjectContext: NSManagedObjectContext! 

    init(context: NSManagedObjectContext) { 
        self.managedObjectContext = context
    }

    viewDidLoad() { 
    self.child = CustomTVC(context: self.managedObjectContext)
    self.child?.view.frame = SomeFrameThatFits
    self.addChild(child!)
    self.addSubview(child!.view)

   //Targeting
   self.someButton.addTarget(self, action: #selector(addHello()), for: .touchUpInside)
   }

   @objc func addHello() { 
    _ = CoreDataWriter(context: self.managedObjectContext).save()
   }
}

太棒了,太棒了,整洁,干得好。我正在使用提供的context 在 SceneDelegate 中创建ViewController 类。

SceneDelegate

...

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

if let windowScene = scene as? UIWindowScene {
            let window = UIWindow(windowScene: windowScene)
                window.rootViewController = ViewController(context: context)
            self.window = window
            window.makeKeyAndVisible()
}

现在,在我的ViewController 课程中,我有一个按钮sendButton。所有这些按钮都会使用单独的 CoreDataWriter 类向我的 CoreData 添加另一个项目,如下所示。

class CoreDataWriter { 
init(context: NSManagedObjectContext) {
    self.managedObjectContext = context
}

func save() {
   let message = Message(context: managedObjectContext)
   message.text = "Hello World!"

     do {
            try self.managedObjectContext.save()
            return nil
       } catch let error {
            print(error.localizedDescription)
       }
    }
}

现在,我 字面意思 copy and pasted Apple's example code 进入我的 Table View Controller 类:

   class CustomTVC: UITableViewController, NSFetchedResultsControllerDelegate {
    var managedObjectContext: NSManagedObjectContext!
    var fetchedResultsController: NSFetchedResultsController<Message>!

    init(context: NSManagedObjectContext) {
        self.managedObjectContext = context
        super.init(nibName: nil, bundle: nil)
    }

    override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
        super.init(nibName: nil, bundle: nil)
    }

    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    override func viewDidLoad() {
        super.viewDidLoad()

        self.tableView.register(UITableViewCell.self, forCellReuseIdentifier: "default")
         initializeFetchedResultsController()
    }


    //MARK: CoreData connections
    func initializeFetchedResultsController() {
        let request = NSFetchRequest<Message>(entityName: "Message")
        let sort = NSSortDescriptor(key: "sent", ascending: true)
        let predicate = NSPredicate(format: "chat.id == %@", "123454")


        fetchedResultsController = NSFetchedResultsController(fetchRequest: request, managedObjectContext: self.managedObjectContext, sectionNameKeyPath: "chat.id", cacheName: "rootCache")
        fetchedResultsController.delegate = self

        do {
            try fetchedResultsController.performFetch()
        } catch {
            fatalError("Failed to initialize FetchedResultsController: \(error)")
        }
    }

    // MARK: - Table view data source

  override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "default", for: indexPath)
        // Set up the cell
        guard let object = self.fetchedResultsController?.object(at: indexPath) else {
            fatalError("Attempt to configure cell without a managed object")
        }
        //Populate the cell from the object
        cell.textLabel?.text = object.text
        return cell
    }

    override func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
         guard let sections = fetchedResultsController.sections else {
                   fatalError("No sections in fetchedResultsController")
               }
               let sectionInfo = sections[section]
               return sectionInfo.numberOfObjects
    }

    //MARK: More stuff for NSData

    func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
        print("controller will change content")
        tableView.beginUpdates()
    }

    func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange sectionInfo: NSFetchedResultsSectionInfo, atSectionIndex sectionIndex: Int, for type: NSFetchedResultsChangeType) {
        print("in ctronoller")
        switch type {
        case .insert:
            tableView.insertSections(IndexSet(integer: sectionIndex), with: .fade)
        case .delete:
            tableView.deleteSections(IndexSet(integer: sectionIndex), with: .fade)
        case .move:
            break
        case .update:
            break
        @unknown default:
            fatalError()
        }
    }

    func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?) {
        print("in didChange controller")
        switch type {
        case .insert:
            tableView.insertRows(at: [newIndexPath!], with: .fade)
        case .delete:
            tableView.deleteRows(at: [indexPath!], with: .fade)
        case .update:
            tableView.reloadRows(at: [indexPath!], with: .fade)
        case .move:
            tableView.moveRow(at: indexPath!, to: newIndexPath!)
        @unknown default:
            fatalError()
        }
    }

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

在我看来,如果我按下方便的花花公子按钮,tableview 应该会用新的“Hello World”刷新 - 但我什么也没得到。没有打印,没有错误。没有。

我已经检查了从 AppDelegate 上下文一直到发送按钮的上下文引用 - 相同的地址。我检查了fetchedResultsController 及其代表 - 不是零。

对于正在发生的事情,我完全、完全、完全不知所措。有什么想法吗?

【问题讨论】:

  • 这不是真正的代码,因为save() 方法没有返回值。并且NSFetchedResultsController 必须有一个排序描述符。
  • CustomTVC 还是 ChatroomTVC?
  • 这可能是无关的,也可能是一切。您需要告诉您的子视图控制器它有一个父级(并告诉父级它有一个子级),以便调用相关的生命周期方法。见this answer
  • @Joakim Daniels CustomTVC,我删除了不相关的代码并忘记编辑名称,很好!
  • @vadian 真正的代码有一个排序描述符,是的,save() 函数在那里有一个额外的回报 - 我截断代码以专注于基本要素但忘记编辑它,很好

标签: ios swift uitableview core-data nsfetchedresultscontroller


【解决方案1】:

如果其他人遇到这个问题 - 我的 Fetch Request 没有正确连接 (?) 到 CoreData。正如@vadian 提到的,NSSortDescriptor必需的。

更重要的是,我在注释掉的浮动谓词是错误的;我在数据结构中比较的id 是一个UUID - 但是,我使用CVarArg 作为String

当我将谓词更改为使用argumentArray 时,所有问题都得到了解决。

所以,改变这个:

        let predicate = NSPredicate(format: "chat.id == %@", "12345")

对此:

        let predicate = NSPredicate(format: "chat.id == %@",  argumentArray: [SomeUUID])

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-06-05
    • 1970-01-01
    • 2019-09-20
    • 2021-05-31
    相关资源
    最近更新 更多