gpt4 book ai didi

ios - NSFetchedResultsController - 不调用委托(delegate)函数

转载 作者:行者123 更新时间:2023-11-29 05:10:40 25 4
gpt4 key购买 nike

我一直在 Stack 上到处寻找答案,但我似乎找不到任何有效的东西。问题很简单 - 我有一个 View Controller ,它有一个子 TableView Controller 。

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进入我的 TableView 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()
}

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

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

我对正在发生的事情感到完全、完全、彻底的迷失。有什么想法吗?

最佳答案

万一其他人遇到这个问题 - 我的 Fetch 请求未正确连接(?)到 CoreData。正如 @vadian 提到的,NSSortDescriptor必需的。

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

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

所以,改变这个:

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

对此:

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

关于ios - NSFetchedResultsController - 不调用委托(delegate)函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59712171/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com