无法让 UITableViewController 动态填充

Posted

技术标签:

【中文标题】无法让 UITableViewController 动态填充【英文标题】:Can't get UITableViewController to dynamically populate 【发布时间】:2015-01-12 01:51:18 【问题描述】:

我正在学习一些关于如何填充表格视图的基本教程。我在 UITableView 和动态填充方面取得了成功,但是对于 UITableViewController 来说,遵循相同的步骤对我不起作用。

任何解释为什么我在使用控制器时遇到困难?

UITableViewController 上的信息似乎没有 UITableView 那么多。为什么每个人似乎都倾向于只使用 UITableView?

为了记录,这是我正在关注的教程:http://www.ioscreator.com/tutorials/tableview-tutorial-in-ios8-with-swift

import UIKit

class TestTableViewController: UITableViewController, UITableViewDelegate 

    let tableData = ["One", "Two", "Three"]

    override func viewDidLoad() 
        super.viewDidLoad()

        // Uncomment the following line to preserve selection between presentations
        // self.clearsSelectionOnViewWillAppear = false

        // Uncomment the following line to display an Edit button in the navigation bar for this view controller.
        // self.navigationItem.rightBarButtonItem = self.editButtonItem()
    

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

    // MARK: - Table view data source

    override func numberOfSectionsInTableView(tableView: UITableView) -> Int 
        // #warning Potentially incomplete method implementation.
        // Return the number of sections.
        return 0
    

    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
        // #warning Incomplete method implementation.
        // Return the number of rows in the section.
        return countElements(tableData)
    


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

        // Configure the cell...
        cell.textLabel?.text = tableData[indexPath.row]

        return cell
    


    /*
    // Override to support conditional editing of the table view.
    override func tableView(tableView: UITableView, canEditRowAtIndexPath indexPath: NSIndexPath) -> Bool 
        // Return NO if you do not want the specified item to be editable.
        return true
    
    */

    /*
    // Override to support editing the table view.
    override func tableView(tableView: UITableView, commitEditingStyle editingStyle: UITableViewCellEditingStyle, forRowAtIndexPath indexPath: NSIndexPath) 
        if editingStyle == .Delete 
            // Delete the row from the data source
            tableView.deleteRowsAtIndexPaths([indexPath], withRowAnimation: .Fade)
         else if editingStyle == .Insert 
            // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
            
    
    */

    /*
    // Override to support rearranging the table view.
    override func tableView(tableView: UITableView, moveRowAtIndexPath fromIndexPath: NSIndexPath, toIndexPath: NSIndexPath) 

    
    */

    /*
    // Override to support conditional rearranging of the table view.
    override func tableView(tableView: UITableView, canMoveRowAtIndexPath indexPath: NSIndexPath) -> Bool 
        // Return NO if you do not want the item to be re-orderable.
        return true
    
    */

    /*
    // MARK: - Navigation

    // In a storyboard-based application, you will often want to do a little preparation before navigation
    override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) 
        // Get the new view controller using [segue destinationViewController].
        // Pass the selected object to the new view controller.
    
    */


将 numberOfSectionsInTableView 更改为返回 1 后,这是我收到的错误。

2015-01-11 18:30:11.557 TableViewTest[30788:8219862] * 断言 -[UITableView 中的失败 dequeueReusableCellWithIdentifier:forIndexPath:], /SourceCache/UIKit_Sim/UIKit-3318.16.14/UITableView.m:6116 2015-01-11 18:30:11.560 TableViewTest[30788:8219862] * 终止应用程序由于 未捕获的异常“NSInternalInconsistencyException”,原因:“无法 使用标识符reuseIdentifier出列一个单元格 - 必须注册一个 nib 或一个类作为标识符或连接一个原型单元 故事板'

【问题讨论】:

您能否为您的表格视图方法添加代码,以便我们看到您在做什么? 刚刚添加。其他人添加了一个答案(似乎已被删除)建议添加 UITableViewDelegate。但是,当我运行时仍然没有填充。 为您的表格视图设置委托和数据源。我认为你应该在 numberOfSectionsInTableView 方法中返回 1 delegate 和 dataSource 都连接到视图控制器。在 numberOfSectionsInTableView 方法中输入 1 并出现错误。进一步调查错误。看起来它在抱怨出队调用中使用的“reuseIdentifier”。 【参考方案1】:

根据您的错误消息,这是解决方案:

在 ViewDidLoad() 中,添加:

self.tableView.registerClass(UITableViewCell.self, forCellReuseIdentifier: "reuseIdentifier")

在此处查看我的示例项目:https://github.com/ericcgu/EGSwiftTableViewController

【讨论】:

谢谢!那是缺失的部分。我刚刚离开并取消了我需要覆盖的函数中的代码的注释。不知道我需要为此添加额外的顶部。 没问题。另外,你也可以通过移除 UITableViewDelegate 来整理你的代码。使用 TableViewController 时不需要这个。如果您只关心一个部分,您也可以删除 didReceiveMemoryWarning 和 numberOfSections,因为它们是可选的。 我不完全理解 UITableViewCell.self 在做什么。我以前从未见过在这样的对象之后自我引用。你能解释一下吗? 当然。目标是注册一个类。这里我们注册了一个普通的 UITableViewCell。或者,我们可以注册一个自定义 TableViewCell 类。 [AnyObject 类] 的 Swift 等价物是 AnyObject.self。这称为类自省。 嗯,好的。我最熟悉 Python 并且(显然)还在学习 Swift,所以我觉得它有点奇怪。我习惯于通过课程而不必事后指定自我。谢谢你的解释!【参考方案2】:

您的异常原因在这里:

let cell = tableView.dequeueReusableCellWithIdentifier("reuseIdentifier", forIndexPath: indexPath) as UITableViewCell

你没有问,但我会提供一些背景信息(以防你想知道为什么会这样)

为了保留内存并提高性能,系统会将屏幕外单元格放入可重复使用的池中。如果 UITableView 需要显示一个单元格,它的数据源会首先检查可重用单元格池。如果有,它(数据源)将组成单元格并将其返回给 UITableView。

因此,本质上,您的数据源对象检查了池中具有 type "reuseIdentifier" 的单元格,但没有找到,因此引发了异常。

您可以使用 UITableView 的 registerClass 方法为重用标识符注册单元格,或者在情节提要或 xib 文件的属性检查器中输入重用标识符。

希望这会有所帮助!

【讨论】:

以上是关于无法让 UITableViewController 动态填充的主要内容,如果未能解决你的问题,请参考以下文章

无法让辅助 UITableViewController 在 UITabBarController 中显示

我无法在 Monotouch UITableViewController 中禁用页眉或页脚

iOS:尝试将导航栏添加到模态 UITableViewController

使用 UITableViewController 时如何让 pickerView 粘在底部?

UIViewController 无法查看 UITableViewController

无法更改 UITableViewController 颜色背景