只有在向CoreData添加对象后才能在bundle中找到NIB?

时间:2018-02-28 00:46:34

标签: swift core-data nib

这让我困扰了几个小时,在查看了各种相关问题之后,我似乎无法找到答案。

我收到一个奇怪的NSInternalInconsistencyException错误。该错误表明它无法加载名为 CountTableViewController 的NIB。但是,当我第一次运行应用程序时(在 模拟器和物理设备中),我可以很好地转向该视图控制器。它加载并看起来就像在 Main.storyboard 文件中一样。然后,当我向后导航并激活另一个视图控制器,将一些测试数据加载到我正在使用的核心数据堆栈时,事情就变得非常糟糕。使用断点和控制台日志,我可以看到单个对象已成功添加到核心数据堆栈中。但是,当我再次点击 CountTableViewController 时,应用程序崩溃并出现以下错误:

*** Terminating app due to uncaught exception 'NSInternalInconsistencyException',
reason: 'Could not load NIB in bundle: 'NSBundle <FILEPATH> (loaded)' with name 
'CountTableViewController''

我已经阅读了我能找到的所有相关问题,这里有一些我已经尝试过的不成功的快速列表:

  • 检查相关文件的拼写/大小写以及它们被引用的任何位置
    • (CountTableViewController.swift,CountTableViewCell.swift,CountReuseCell [reuse id])
  • 删除对相关文件的引用,重新拖入Xcode
  • 删除对Main.storyboard和LaunchScreen.storyboard的引用,重新拖入Xcode
  • 验证项目的“构建阶段”中的“编译源”下是否正确列出了每个文件
  • 验证故事板是否在项目的“构建阶段”中的“复制捆绑资源”下正确列出
  • 从头开始重写为全新的视图控制器(仍有相同的错误)

非常确定它与CoreData有关,因为这似乎是视图控制器执行和不工作之间的唯一区别,但我对Swift和iOS dev很新,所以我可能会偏离目标。

我有堆栈跟踪,我将在下面发布。我将尝试发布一些我认为有用的代码的简洁部分。提前感谢您的帮助!

堆栈跟踪:

0   CoreFoundation                      0x000000010721912b __exceptionPreprocess + 171
1   libobjc.A.dylib                     0x0000000102a78f41 objc_exception_throw + 48
2   CoreFoundation                      0x000000010728e245 +[NSException raise:format:] + 197
3   UIKit                               0x00000001043fd098 -[UINib instantiateWithOwner:options:] + 501
4   UIKit                               0x00000001040b2687 -[UITableView _dequeueReusableViewOfType:withIdentifier:] + 590
5   UIKit                               0x0000000120c4f79d -[UITableViewAccessibility dequeueReusableCellWithIdentifier:] + 147
6   UIKit                               0x00000001040b2b6b -[UITableView _dequeueReusableCellWithIdentifier:forIndexPath:usingPresentationValues:] + 148
7   UIKit                               0x00000001040b2aa3 -[UITableView dequeueReusableCellWithIdentifier:forIndexPath:] + 89
8   UIKit                               0x0000000120c4f8e0 -[UITableViewAccessibility dequeueReusableCellWithIdentifier:forIndexPath:] + 285
9   BSA Inventory Control               0x00000001018d35d6 _T021BSA_Inventory_Control24CountTableViewControllerC05tableF0So07UITableF4CellCSo0iF0C_10Foundation9IndexPathV12cellForRowAttF + 774
10  BSA Inventory Control               0x00000001018d3c4c _T021BSA_Inventory_Control24CountTableViewControllerC05tableF0So07UITableF4CellCSo0iF0C_10Foundation9IndexPathV12cellForRowAttFTo + 92
11  UIKit                               0x00000001040ce484 -[UITableView _createPreparedCellForGlobalRow:withIndexPath:willDisplay:] + 778
12  UIKit                               0x00000001040cea2a -[UITableView _createPreparedCellForGlobalRow:willDisplay:] + 74
13  UIKit                               0x00000001040941f6 -[UITableView _updateVisibleCellsNow:isRecursive:] + 3031
14  UIKit                               0x00000001040b62e6 -[UITableView layoutSubviews] + 176
15  UIKit                               0x000000010403ea6d -[UIView(CALayerDelegate) layoutSublayersOfLayer:] + 1439

CountTableViewController.swift

import UIKit
import CoreData
class CountTableViewController: UITableViewController {
    let coreDataStack = CoreDataStack.shared
    var moc: NSManagedObjectContext! = nil
    var counts: [Count] = []

    override func viewDidLoad() {
        super.viewDidLoad()
        moc = coreDataStack.viewContext

        tableView.register(CountTableViewCell.self, forCellReuseIdentifier: "CountReuseCell")
        let nib = UINib(nibName: "CountTableViewController", bundle: nil)
        tableView.register(nib, forCellReuseIdentifier: "CountReuseCell")
    }
    func reload() {
        counts = Count.items(for: moc, matching: nil, sortedBy: nil)
    }
    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        reload()
        tableView.reloadData()
    }
    override func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }
    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return counts.count
    }
    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        guard let cell = tableView.dequeueReusableCell(withIdentifier: "CountReuseCell", for: indexPath) as? CountTableViewCell else  {
            fatalError("The dequeued cell is not an instance of CountTableViewCell")
        }
        let count = counts[indexPath.row]
        cell.nameLabel.text = count.name       
        return cell
    }
}

CountTableViewCell.swift 文件只是一个默认的Cocoa Touch类,其子类为UITableViewCell,单个插座链接到故事板中的nameLabel

一些快速的事后......如果有任何我需要的信息,请告诉我。我正在使用核心数据堆栈类和一个由我在一年前开发的iOS开发类教授改编的扩展程序,它提供了易于使用的.items()功能......如果您认为这样,我也可以发布造成任何问题。这很奇怪,因为我有另一个核心数据实体(Product而不是Count)使用几乎完全相同的完全相同的代码。我似乎无法弄清楚这种情况有什么不同......

Xcode版本9.2(9C40b),iOS版本11.2(15C107)

1 个答案:

答案 0 :(得分:1)

只是为了把事情包装起来并希望将来帮助某人,这个问题似乎已经出现在 CountTableViewController.swift 中的这些行中:

tableView.register(CountTableViewCell.self, forCellReuseIdentifier: "CountReuseCell")
let nib = UINib(nibName: "CountTableViewController", bundle: nil)
tableView.register(nib, forCellReuseIdentifier: "CountReuseCell")

像往常一样,这似乎是重用旧代码和复制/粘贴的混合物。奇怪的是,这个代码在同一个项目中与Product实体几乎完全相同的情况下似乎工作得很好。尽管如此,我删除了上面的最后两行,只留下了:

tableView.register(CountTableViewCell.self, forCellReuseIdentifier: "CountReuseCell")

果然, NSInternalInconsistencyException 错误消失了!

昨晚,我能够完全以编程方式重写 CountTableViewController 并且工作正常,所以我打算暂时使用它。但是,很高兴知道未来导致此错误的原因。

感谢@pbasdf的评论/回答!