NSFetchedResultsController 返回行并填充表格,但表格单元格标签消失

Posted

技术标签:

【中文标题】NSFetchedResultsController 返回行并填充表格,但表格单元格标签消失【英文标题】:NSFetchedResultsController returns row and populates table, but the table cell labels disappear 【发布时间】:2016-07-20 06:23:29 【问题描述】:

我遇到了 NSFetchedResultsViewController 的问题。这是我的设置。我的应用程序 rootViewController 是 TabBarController。连接到我的 UITabBarController 的是两个视图控制器(所以我有 2 个选项卡)。与每个选项卡关联的 UIViewControllers 都有我嵌入另一个 UIViewController 的容器。其中一个 UIViewControllers 有一个 UITableView,我试图用来自 CoreData 的数据填充它。所以,为了填充 UITableView,我使用 NSFetchedResultsViewController。

这是我的奇怪问题。如果我使用 UITableView / NSFetchedResultsController 将 UIViewController 设为第一个选项卡,以便在应用程序加载时首先显示它,那么一切正常,并且我在 tableview 中看到了我的 3 行数据(我在每个自定义中设置了两个标签)表格单元格)。

Labels visible for each row when on first tab

但是,如果我将该 ViewController 作为第二个选项卡,启动应用程序,查看我的第一个选项卡,然后点击我的 tableview 所在的第二个选项卡,创建 3 行,但我看不到标签,只是一个带有披露指示符的空白行。

Labels disappear, but the rows are still created when on second tab

NSFectchedResultsController 似乎工作正常,因为即使在缺少单元格标签但创建行的情况下,我也会注销应该显示的值,并打印值,所以我的 managedobject 具有值.

以前我没有使用 CoreData,只是在内存中用简单的对象填充表格,在标签之间点击时一切正常……数据总是出现。我是在实现了 fetchedResultsController 之后才开始看到这个问题的。

这是我的代码:

class OrderListViewController: UIViewController, UITableViewDelegate, UITableViewDataSource, NSFetchedResultsControllerDelegate 

    @IBOutlet weak var searchBar: UISearchBar!
    @IBOutlet weak var tableView: UITableView!

    var delegate: OrderListViewControllerDelegate?

    private lazy var fetchedResultsController: NSFetchedResultsController = 
        let request = NSFetchRequest(entityName: "ListOrder")
        let dateSort = NSSortDescriptor(key: "statusSubmittedAt", ascending: true)
        request.sortDescriptors = [dateSort]

        let pred = NSPredicate(format: "(statusBuilt == %@ and statusCancelled == %@ and statusComplete == %@ and statusSubmitted == %@)", NSNumber(bool: false), NSNumber(bool: false), NSNumber(bool: false), NSNumber(bool: true))
        request.predicate = pred

        let moc = (UIApplication.sharedApplication().delegate as? AppDelegate)!.managedObjectContext
        let controller = NSFetchedResultsController(fetchRequest: request, managedObjectContext: moc, sectionNameKeyPath: nil, cacheName: nil)
        controller.delegate = self

        return controller
    ()

    override func viewDidLoad() 
        super.viewDidLoad()

        // Do any additional setup after loading the view.
        do 
            try fetchedResultsController.performFetch()
         catch 
            fatalError("Failed to initialize FetchedResultsController: \(error)")
        
    

    override func didReceiveMemoryWarning() 
        super.didReceiveMemoryWarning()
        //Dispose of any resources that can be recreated.
    

    func numberOfSectionsInTableView(tableView: UITableView) -> Int 
        return fetchedResultsController.sections!.count
    

    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
        let sections = fetchedResultsController.sections! as [NSFetchedResultsSectionInfo]
        let sectionInfo = sections[section]
        return sectionInfo.numberOfObjects
    

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell 
        let cell = (tableView.dequeueReusableCellWithIdentifier("OrderListTableViewCell", forIndexPath: indexPath) as? OrderListTableViewCell)!

        // Configure the cell...
        let order = (fetchedResultsController.objectAtIndexPath(indexPath) as? ListOrder)!

        // Update Cell
        if let userDisplayName = order.userDisplayName 
            print(userDisplayName) //prints correct value to console
            cell.orderName.text = userDisplayName
        

        if let placementDescription = order.tagPlacementDescription 
            cell.orderName.text = cell.orderName.text! + " - \(placementDescription)"
            print(placementDescription) //prints correct value to console

        

        if let orderTime = order.statusSubmittedAt 
            cell.orderTime.text = String.stringTimeFromDate(orderTime)
        

        return cell
    


    func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) 

        self.delegate?.didSelectOrder(self, order:(fetchedResultsController.objectAtIndexPath(indexPath) as? ListOrder)!)
    

非常感谢任何帮助。我没有想法。

【问题讨论】:

如何在容器视图中嵌入带有表格视图的控制器?标签是否存在但大小为零或超出屏幕范围? 我遇到了同样的问题,这通常发生在我从推送的视图控制器向我的核心数据添加新项目并导航回来时。你找到解决办法了吗? 【参考方案1】:

您的代码似乎是正确的,但我宁愿为您提供 NSFetchResultsController 通常在代码中实现的方式。您需要在您的委托中实现至少三个方法

    controllerWillChangeContent: controllerDidChangeContent: controller:DidChangeObject:atIndexPath:forChangeType:newIndexPath

希望这会有所帮助。

P。 S.您确定每次打印的值都作为单元格的方法被调用。尝试设置断点或 else-clause 去检查这个。此外,我会重新检查单元的标识符和插座的连接

【讨论】:

我实际上也实现了这些方法,但已将它们注释掉以查看它们是否是罪魁祸首(并将它们从我发布的代码中删除)。如果我用 NSFetchResultsController 读入数据并且数据没有改变,那么根据我的理解,这些委托方法不会被调用。无论是否实施,我都会遇到同样的问题。是的,每次调用 cellForRowAtIndexPath 时都会打印出我的值。我相信我的标识符和插座也是正确的,因为当我的视图控制器位于第一个选项卡上时,表格单元格会正确显示。 好吧,你能把你的项目上传到任何地方,这样我就可以下载并稍后检查它是否有错误?我发现这个错误很有趣

以上是关于NSFetchedResultsController 返回行并填充表格,但表格单元格标签消失的主要内容,如果未能解决你的问题,请参考以下文章

在 Core Data 应用程序中调用 performFetch 后,是不是需要手动更新表视图?