Swift prepareForSegue对象为nil

时间:2017-03-13 18:28:32

标签: ios swift object null

我试图将事件对象传递给DetailViewController,但它崩溃并说:"致命错误:在解开一个Optional值"时意外地发现了nil。我很肯定错误是在我的准备(对于segue :)功能,但我已经在这几个小时,我无法找到如何解决它。任何帮助,将不胜感激。谢谢!

这是我的代码:

class MainViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {

@IBOutlet weak var textView: UITextView!
@IBOutlet weak var tableView: UITableView!

var events = [Event]()

override func viewDidLoad() {
    super.viewDidLoad()

    // Do any additional setup after loading the view.


    tableView.delegate = self
    tableView.dataSource = self

    parseCSV()
}

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


@IBAction func logOutAction(sender: AnyObject) {
    if FIRAuth.auth()?.currentUser != nil {
        do {
            try FIRAuth.auth()?.signOut()
            let vc = UIStoryboard(name: "Main", bundle: nil).instantiateViewController(withIdentifier: "SignUp")
            present(vc, animated: true, completion: nil)

        } catch let error as NSError {
            print(error.localizedDescription)
        }
    }
}

func parseCSV() {
    let path = Bundle.main.path(forResource: "eventData", ofType: "csv")
    do {
        let csv = try CSV(contentsOfURL: path!)
        let rows = csv.rows

        for row in rows {
            let eventTitle = row["Title "]!
            let eventLoc = row[" Location "]!
            let eventStart = row[" Start_Time "]!
            let eventEnd = row [" End_Time"]!

            let event = Event(title: eventTitle, loc: eventLoc, start: eventStart, end: eventEnd)
            events.append(event)
        }
    } catch let err as NSError {
        print(err.debugDescription)
    }
}


func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    if let cell = tableView.dequeueReusableCell(withIdentifier: "eventCell", for: indexPath) as? EventCell{
        let event = events[indexPath.row]
        cell.updateUI(event: event)

        return cell
    } else {
        return UITableViewCell()
    }
}

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return events.count
}

func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
    let event = events[indexPath.row]
    performSegue(withIdentifier: "detailSegue", sender: event)
}

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "detailSeque" {
        if let detailVC = segue.destination as? DetailViewController {
            if let event = sender as? Event {
                detailVC.event = event
            }
        }
    }
}
}

我的DetailViewController类:

class DetailViewController: UIViewController {

var event: Event!

@IBOutlet weak var eventTitle: UILabel!

override func viewDidLoad() {
    super.viewDidLoad()

    eventTitle.text = event.title // error here. says event is nil.
    }
}

1 个答案:

答案 0 :(得分:1)

由于segue已从您的单元格连接,因此当您点击单元格时,它已自动执行。以编程方式执行会导致您的segue执行两次:一次正确传递值,一次没有传递。

您可以更改要从视图控制器连接的segue而不是当前代码可以使用的单元格,或更新代码以执行prepareForSegue中的所有逻辑并删除您的实现didSelectRowAt

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "detailSeque" {
        if let detailVC = segue.destination as? DetailViewController {
           let row = tableView.indexPathForSelectedRow!.row
           detailVC.event = events[row]
        }
    }
}