swift - 核心数据: Showing NSFetchRequestResult Search Results with UITableView

标签 swift uitableview core-data nsfetchedresultscontroller

过去几天我一直在玩Core Data。我正在使用 NSFetchedResultsController 获取数据。我的实体具有年龄(Int)、firstName(字符串)、lastName(字符串)、uuid(字符串)等属性。我能够将新记录插入数据库。我可以删除一条记录。我还可以编辑记录。我无法做的是用表格显示搜索结果。

class HomeViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {
    // MARK: - Instance variables
    private let persistentContainer = NSPersistentContainer(name: "Profiles") // core data model file (.xcdatamodeld)
    var managedObjectContext: NSManagedObjectContext?

    // MARK: - IBOutlets
    @IBOutlet weak var tableView: UITableView!
    @IBOutlet weak var searchButton: UIButton!

    // MARK: - IBActions
    @IBAction func searchTapped(_ sender: UIButton) {
        searchRecords()
    }

    // MARK: - Life cycle
    override func viewDidLoad() {
        super.viewDidLoad()

        // loading persistentContainer //
        persistentContainer.loadPersistentStores { (persistentStoreDescription, error) in
            if let error = error {
                print("Unable to Load Persistent Store")
                print("\(error), \(error.localizedDescription)")
            } else {
                do {
                    try self.fetchedResultsController.performFetch()
                } catch {
                    let fetchError = error as NSError
                    print("Unable to Perform Fetch Request")
                    print("\(fetchError), \(fetchError.localizedDescription)")
                }
            }
        }
    }
    // MARK: - Life cycle

    // MARK: - fetchedResultsController(controller with the entity)
    fileprivate lazy var fetchedResultsController: NSFetchedResultsController<Person> = {
        // Create Fetch Request with Entity
        let fetchRequest: NSFetchRequest<Person> = Person.fetchRequest()

        // Configure Fetch Request
        fetchRequest.sortDescriptors = [NSSortDescriptor(key: "lastName", ascending: true)]

        // Create Fetched Results Controller
        let fetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: self.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)

        // Configure Fetched Results Controller
        fetchedResultsController.delegate = self
        return fetchedResultsController
    }()
    // MARK: - fetchedResultsController

    // MARK: - TableView
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        guard let people = fetchedResultsController.fetchedObjects else { return 0 }
        return people.count
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! ProfileTableViewCell
        let person = fetchedResultsController.object(at: indexPath)

        // Configure Cell
        cell.firstLabel.text = person.firstName
        cell.lastLabel.text = person.lastName
        cell.ageLabel.text = String(person.age)
        return cell
    }
    // MARK: - TableView

    // MARK: - Showing search result
    func searchRecords() {
        let context = self.persistentContainer.viewContext
        let fetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "Person")
        let predicate = NSPredicate(format: "firstName CONTAINS[c] %@", "Sandra")
        fetchRequest.predicate = predicate
        do {
            try fetchedResultsController.performFetch()
            /*
            let result = try context.fetch(fetchRequest)
            if (result.count > 0) {
                print(result.count)
            }
            */
        } catch {
            print("bad")
        }
    }
    // MARK: - Showing search result
}

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

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

    func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?) {
        switch (type) {
        case .insert:
            if let indexPath = newIndexPath {
                tableView.insertRows(at: [indexPath], with: .fade)
            }
            break;
        case .delete:
            if let indexPath = indexPath {
                tableView.deleteRows(at: [indexPath], with: .fade)
            }
            break;
        case .update:
            if let indexPath = indexPath {
                tableView.reloadRows(at: [indexPath], with: .automatic)
            }
            break;
        default:
            tableView.reloadData()
        }
    }

    func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange sectionInfo: NSFetchedResultsSectionInfo, atSectionIndex sectionIndex: Int, for type: NSFetchedResultsChangeType) {
    }
}

如果我运行 searchRecords()result.count 将正确返回记录数。但 table 保持不变。那么如何用表格显示我的搜索结果呢?谢谢。

最佳答案

将您的表格 View 数据源替换为

// MARK: - Table View

override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
    return self.fetchedResultsController.sections?.count ?? 0
}

override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    let sectionInfo = self.fetchedResultsController.sections![section]
    return sectionInfo.numberOfObjects
}

希望对你有帮助

关于swift - 核心数据: Showing NSFetchRequestResult Search Results with UITableView,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48970635/

相关文章:

ios - 当原型(prototype)单元不工作时如何使用 segue

iOS UITableView 隐藏表头空间

ios - 无法将数据从 UITableviewController 传递到 ViewController - 不断收到 Swift 动态 Cast 类无条件错误

ios - 试验 Firebase - 推送通知

ios - 如果 UITableView 的单元格分为多个部分,如何将数据从 UITableView 传输到另一个 View Controller ?

swift - 可以快速将自定义 HTML View 加载到 webView 中吗?

iOS MVC : get view controller class from model?

ios - 设置新的核心数据实体属性值不保存。这与 RestKit 有什么关系吗?

ios - Swift/从 tableView 执行自定义转场

swift - 如何在 Swift 中实现 "abstract"属性初始值设定项