微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

可选的致命错误没有意义

如何解决可选的致命错误没有意义

我不断收到一个致命错误,说一个值是如何解包的,它是 nil,我不明白是如何解包的。当我使用特定变量实例化一个视图控制器时,它们都会显示出来,但是当我对确切的 VC 执行 segue 时,这些值不会显示出来。

以这些函数为例...

override func tableView(_ tableView: UITableView,didSelectRowAt indexPath: IndexPath) {
    tableView.deselectRow(at: indexPath,animated: true)
    
  
    
    if let displayVC = storyboard?.instantiateViewController(withIdentifier: Constants.Storyboards.TeachStoryboardID) as? SchoolEventDetailsViewController {
        
        displayVC.selectedEventName = events[indexPath.row].eventName
        displayVC.selectedEventDate = documentsDate[indexPath.row].eventDate
        displayVC.selectedEventCost = documentsCost[indexPath.row].eventCost
        displayVC.selectedEventGrade = documentsGrade[indexPath.row].eventGrade
        displayVC.selectedEventDocID = documentsID[indexPath.row]?.docID
        
        navigationController?.pushViewController(displayVC,animated: true)
    }
}

结合这个功能

func verifyInstantiation() {
    if let datetoLoad = selectedEventDate {
        dateEditableTextF.text = datetoLoad
    }
    
    if let costToLoad = selectedEventCost {
        costEditableTextF.text = costToLoad
    }
    
    if let gradesToLoad = selectedEventGrade {
        gradesEditableTextF.text = gradesToLoad
    }
    
    if let docIDtoLoad = selectedEventDocID {
        docIDUneditableTextF.text = docIDtoLoad
    }
    
    if let eventNametoLoad = selectedEventName {
        eventNameEditableTextF.text = eventNametoLoad
    }
}

帮助完美加载数据,但是当我尝试从搜索控制器执行 segue 时,数据不存在。

override func viewDidLoad() {
    super.viewDidLoad()
    
    navigationItem.title = selectedEventName

我将 vc 的标题设置为具有事件名称,并且我最近还添加一个文本字段来存储它以用于实验目的(这个问题)。

现在的问题是我想将数据从 Algolia 搜索控制器传输到该 VC,并且我让所有其他字段都显示出来,除了一个是文档 ID。因此,我创建了一个完成处理函数获取文档 ID 作为字符串,并在执行 segue 时将其插入到 vc 中,就像在实例化 vc 时一样。

这是函数

func getTheEventDocID(completion: @escaping ((String?) -> ())) {
    documentListener = db.collection(Constants.Firebase.schoolCollectionName).whereField("event_name",isEqualTo: selectedEventName ?? navigationItem.title).addSnapshotListener(includeMetadataChanges: true) { (querySnapshot,error) in
        if let error = error {
            print("There was an error fetching the documents: \(error)")
        } else {
            self.documentsID = querySnapshot!.documents.map { document in
                return EventDocID(docID: (document.documentID) as! String)
            }
            let fixedID = "\(self.documentsID)"
            let substrings = fixedID.dropFirst(22).dropLast(3)
            let realString = String(substrings)
            completion(realString)
        }
    }
}

我认为 selectedEventNamenavigationItem.title 可以完成工作并在我现在将展示的数据传输函数中使用该函数时提供值:

 //MARK: - Data Transfer From Algolia Search to School Event Details
override func prepare(for segue: UIStoryboardSegue,sender: Any?) {

    
    otherVC.getTheEventDocID { (eventdocid) in
        if let id = eventdocid {
            if segue.identifier == Constants.Segues.fromSearchToSchoolEventDetails {
                let vc = segue.destination as! SchoolEventDetailsViewController
                vc.selectedEventName = self.nameTheEvent
                vc.selectedEventDate = self.dateTheEvent
                vc.selectedEventCost = self.costTheEvent
                vc.selectedEventGrade = self.gradeTheEvent
                vc.selectedEventDocID = id
            }
        }
    }
 
    
}

但是当点击搜索结果时它最终什么也没显示,这非常令人沮丧,我不明白为什么当我在 SchoolEventDetailsVC 中声明它们时它们都是空值。我试图强制解包 selectedEventName 并且它崩溃说有一个 nil 值,我不知道为什么。这个问题实际上还有很多,但我只是尽量保持简短,以便人们实际上会尝试阅读它并提供帮助,因为没有人会阅读我发布的问题,所以提前致谢。

解决方法

我有点困惑 otherVC 是什么,它在 getTheEventDocID 中设置自己的属性,而在闭包中设置 self 的属性,这是一个不同的控制器。不过没关系,我希望你知道自己在做什么。

由于 getTheEventDocID 异步运行,视图将在数据可用之前加载和显示。因此,viewDidLoad 看不到实际数据,而是很快就会过时的数据。

因此,您需要通知详细信息视图控制器有新数据可用,并刷新其用户界面。类似的东西

override func prepare(for segue: UIStoryboardSegue,sender: Any?) {
    otherVC.getTheEventDocID { (eventdocid) in
        if let id = eventdocid {
            if segue.identifier == Constants.Segues.fromSearchToSchoolEventDetails {
                let vc = segue.destination as! SchoolEventDetailsViewController
                vc.selectedEventName = self.nameTheEvent
                vc.selectedEventDate = self.dateTheEvent
                vc.selectedEventCost = self.costTheEvent
                vc.selectedEventGrade = self.gradeTheEvent
                vc.selectedEventDocID = id
                vc.updateUI()
            }
        }
    }
}

在目标视图控制器中:

class SchoolEventDetailsViewController ... {
    override func viewDidLoad() {
        super.viewDidLoad()
        updateUI()
    }

    func updateUI () {
        navigationItem.title = selectedEventName
        // and so on
    }
}
,

好的,所以我决定尝试一种解决方法并完全放弃 getTheEventDocID() 方法,因为它只会给我带来压力。所以我决定放弃 Firebase 生成的文档 IDS,只使用我制作的函数中的 10 位生成的 id。我还想出了如何通过将随机的 10 位 id 存储在一个变量中并在两个地方使用它来在 Algolia 记录中添加完全相同的 10 位 id。因此,现在我不再使用查询调用来获取 Firebase 生成的文档 ID 并在每次单击搜索结果时让我的应用程序崩溃,而是基本上编辑了 Algolia 记录的 Struct 并添加了一个 eventDocID 属性可以与 hits.hitSource(at: indexPath.row).eventDocID 一起使用。

现在就像我通过 segue 数据传输将其他字段添加到 vc 一样,我现在可以用我的文档 ID 做同样的事情,因为一切都匹配:)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。