在Base ViewController中展开Optional值时意外地发现了nil

时间:2015-07-12 17:37:36

标签: ios swift

我使用pagemenu项目https://github.com/uacaps/PageMenu在我的应用程序中包含分段控件以在视图控制器之间切换,此控件包含一个基本视图控制器(PageMenuViewController1)和两个子视图控制器(ViewController1,ViewController2) )但是当我启动这个PageMenuViewController1时,应用程序在子视图控制器中失败(下面我确切地显示了应用程序失败的位置)并抛出了这个错误:

fatal error: unexpectedly found nil while unwrapping an Optional value
在下面的

中,您可以看到viewcontrollers的代码。

基本视图控制器是:

class PageMenuViewController1: UIViewController {

  var pageMenu : CAPSPageMenu?

  override func viewDidLoad() {
    // Array to keep track of controllers in page menu
    var controllerArray : [UIViewController] = []

    // Create variables for all view controllers you want to put in the
    // page menu, initialize them, and add each to the controller array.
    // (Can be any UIViewController subclass)
    // Make sure the title property of all view controllers is set
    // Example:
    var controller1 : ViewController1 = ViewController1()
    controller1.title = "ViewController1"
    controllerArray.append(controller1)
    var controller2 : ViewController2 = ViewController2()
    controller2.title = "ViewController2"
    controllerArray.append(controller2)

    // Customize page menu to your liking (optional) or use default settings by sending nil for 'options' in the init
    // Example:
    var parameters: [CAPSPageMenuOption] = [
      .MenuItemSeparatorWidth(4.3),
      .UseMenuLikeSegmentedControl(true),
      .MenuItemSeparatorPercentageHeight(0.1)
    ]

    // Initialize page menu with controller array, frame, and optional parameters
    pageMenu = CAPSPageMenu(viewControllers: controllerArray, frame: CGRectMake(0.0, 0.0, self.view.frame.width, self.view.frame.height), pageMenuOptions: parameters)

    // Lastly add page menu as subview of base view controller view
    // or use pageMenu controller in you view hierachy as desired
    self.view.addSubview(pageMenu!.view)
  }
}

子视图控制器:

- ViewController1

class ViewController1: UIViewController, UITableViewDataSource{

  @IBOutlet var tableview:UITableView!

  let apiClient = ApiClient()

  var clubs: [Club]!

  override func viewDidLoad() {
    super.viewDidLoad()

    apiClient.Service.getList() { clubs, error in
      if clubs != nil {
        self.clubs = clubs
        self.tableview.reloadData()// Here is where the lldb throws the exception
      }
      else {
        println("error: \(error)")
      }
    }
  }

  func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return self.clubs?.count ?? 0
  }

  func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) ->UITableViewCell {
    var cell = tableView.dequeueReusableCellWithIdentifier("clubObjectCell") as! ClubTableViewCell
    cell.clubObject = self.clubs?[indexPath.row]
    return cell
  }

}

- ViewController2

class ViewController2: UIViewController, UITableViewDataSource {

  @IBOutlet var tableview:UITableView!

  let apiClient = ApiClient()

  var parties: [Party]? = []

  override func viewDidLoad() {

    super.viewDidLoad()

    apiClient.partiesService.getList() { parties, error in
      if parties != nil {
        self.parties = parties
        self.tableview.reloadData()// Here is where the lldb throws the exception
      }
      else {
        println("error: \(error)")
      }
    }
  }

  func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return self.parties?.count ?? 0
  }

  func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    var cell = tableView.dequeueReusableCellWithIdentifier("partyObjectCell") as! PartyTableViewCell
    cell.partyObject = self.parties?[indexPath.row]
    return cell
  }

}

我希望我的问题很清楚:)

2 个答案:

答案 0 :(得分:1)

最有可能的是,您忘记将tableView IBOutlet属性与故事板中的视图相关联。在排水沟中,在@IBOutlet var tableview: UITableView!代码行的左侧,您应该看到一个圆圈。如果圆圈已填满,则插座已连接。但如果圆圈为空,则需要连接它。

这种联系可以通过几种不同的方式进行。常用方法是使用Assistant Editor。在一个窗格中打开ViewController1.swift,然后在另一个窗格中打开Main.storyboard。在故事板中选择适当的UITableView。然后,按住Ctrl键拖动到IBoutletViewController1附近的空白圈。

如果你的插座已经连接(你可以看到一个完整的圆圈),那么你应该尝试将reloadData()电话转移到viewWillAppear(:),如下所示:

override func viewWillAppear(animated: Bool) {
    super.viewWillAppear(animated)

    apiClient.Service.getList() { clubs, error in
        if clubs != nil {
            self.clubs = clubs
            self.tableview.reloadData()// Here is where the lldb throws the exception
        }
        else {
            println("error: \(error)")
        }
    }
}

答案 1 :(得分:1)

我认为实例化ViewController1和ViewController2的方式不正确。

如果您使用的是故事板,则必须按如下方式实例化:

var controller1 : ViewController1 = storyboard.instantiateViewControllerWithIdentifier("<view controller 1 identifier in storyboard>")

如果您没有使用storyboard,则必须使用NSBundle.mainModule()。loadNibNamed(...)进行实例化。

只是调用类初始值设定项不会将它们附加到storyboard或xib。