ios - Swift 不检索 Firebase 值(返回值作为 NSNull)

标签 ios swift firebase firebase-realtime-database nsnull

我正在尝试从 Firebase 实时数据库中提取值,但无论出于何种原因,这些值都会以 NSNull 的形式返回。我在第 13 行收到错误“无法将类型‘NSNull’的值转换为‘NSNumber’,直到我将‘let count = ...’更改为‘if let’。我不确定为什么这些值返回 null,而实际上它们不是。

我已经尝试打印检索到的快照,并且正如我所预期的那样,没有一个值为空。我很困惑。 这是打印检索到的快照时得到的日志:

[Snap (A12-1A) 0, Snap (B1-2A) 0, Snap (C2-3A) 0, Snap (D3-4A) 0, Snap (E4-5A) 0, Snap (F5-6A) 0, Snap (G6-7A) 0, Snap (H7-8A) 0, Snap (I8-9A) 0, Snap (J9-10A) 5, Snap (K10-11A) 0, Snap (L11-12P) 0, Snap (M12-1P) 0, Snap (N1-2P) 0, Snap (O2-3P) 0, Snap (P3-4P) 0, Snap (Q4-5P) 0, Snap (R5-6P) 0, Snap (S6-7P) 0, Snap (T7-8P) 0, Snap (U8-9P) 0, Snap (V9-10P) 0, Snap (W10-11P) 0, Snap (X11-12A) 0]

此快照中的所有数据都是我的 Firebase 数据库中的数据,因此我不确定为什么会收到此错误。

我的 Firebase 设置:日历注册 --- 04132019 ---(A12-1A:0、B1-2A:0、...、X11-12P:0)

func getCalendarSignupsOnDate(selectedDate: String, handler: @escaping (_ dateSignups: [Calendar]) -> ()) {
        var calendarArray = [Calendar]()

        REF_CALENDAR_SIGNUPS.observeSingleEvent(of: .value) { (snapshot) in
            guard let snapshot = snapshot.children.allObjects as? [DataSnapshot] else { return }

            var doesContainDate = false
            for date in snapshot {
                if date.key == selectedDate {
                    doesContainDate = true

                    self.REF_CALENDAR_SIGNUPS.child(date.key).observeSingleEvent(of: .value, with: { (dateSnapshot) in
                        guard let dateSnapshot = dateSnapshot.children.allObjects as? [DataSnapshot] else { return }

                        for timeSlots in dateSnapshot {
                            let time = timeSlots.key
                            if let count = timeSlots.childSnapshot(forPath: "J9-10A").value as? Int {
                                let calendarData = Calendar(time: time, signupCount: count)
                                calendarArray.append(calendarData)
                                handler(calendarArray)
                            } else {
                                let count = 0
                                let calendarData = Calendar(time: time, signupCount: count)
                                calendarArray.append(calendarData)
                                handler(calendarArray)
                            }
                        }
                    })
                }
            }
            for item in calendarArray {
                print(item.time + " " + String(item.signupCount))
            }
            if doesContainDate == false {
                let dateSignUpData = [
                    "A12-1A": 0,
                    "B1-2A": 0,
                    "C2-3A": 0,
                    "D3-4A": 0,
                    "E4-5A": 0,
                    "F5-6A": 0,
                    "G6-7A": 0,
                    "H7-8A": 0,
                    "I8-9A": 0,
                    "J9-10A": 0,
                    "K10-11A": 0,
                    "L11-12P": 0,
                    "M12-1P": 0,
                    "N1-2P": 0,
                    "O2-3P": 0,
                    "P3-4P": 0,
                    "Q4-5P": 0,
                    "R5-6P": 0,
                    "S6-7P": 0,
                    "T7-8P": 0,
                    "U8-9P": 0,
                    "V9-10P": 0,
                    "W10-11P": 0,
                    "X11-12A": 0
                    ]
                    as [String : Int]

                self.REF_CALENDAR_SIGNUPS.child(selectedDate).updateChildValues(dateSignUpData)

                for timeSlots in dateSignUpData {
                    let time = timeSlots.key
                    let count = timeSlots.value
                    let calendarData = Calendar(time: time, signupCount: count)
                    calendarArray.append(calendarData)
                }
            }

            handler(calendarArray)
        }
    }

最佳答案

这并不是一个完整的答案,但也许可以为您提供正确的方向。代码中有一些令人困惑的部分,因此我将审查这些部分,然后可能提供更好的方向...

您的函数已通过单个选定日期

func getCalendarSignupsOnDate(selectedDate: String

这意味着您知道要读取的日期节点。但是,代码正在读取所有日期并迭代它们以尝试找到该日期

 for date in snapshot {
   if date.key == selectedDate {

直接读取该节点,然后使用其中的子节点可能会少得多的代码。

此外,在代码中,您将迭代 A12-1A、B1-2A 的子节点,查找特定节点

if let count = timeSlots.childSnapshot(forPath: "J9-10A").value as? Int

这根本不起作用,因为 timeSlots A12-1A 等的值全部为 0,而不是子快照。代码继续,显示您想要读取该特定子节点的值,然后将其余部分设置为 0,并将其添加到您的日历数组中。

let count = 0
let calendarData = Calendar(time: time, signupCount: count)

但是再往下看,如果找不到那个时隙,所有子节点无论如何都会设置为 0。

if doesContainDate == false {
    let dateSignUpData = [
       "A12-1A": 0,
       "B1-2A": 0,

所以...我想我明白了你想要做的事情的要点。这里有一些代码可能让您朝着正确的方向前进。这应该替换您问题中的所有代码。

func getCalendarSignupsOnDate(selectedDate: String) {
    var dateSignUpData = [
        "A12-1A": 0,
        "B1-2A": 0,
        "C2-3A": 0,
        "J9-10A": 0
    ]
    let slotYourLookingFor = "J9-10A"
    let dateRef = self.REF_CALENDAR_SIGNUPS.child(selectedDate).child(slotYourLookingFor)
    dateRef.observeSingleEvent(of: .value, with: { snapshot in
        if snapshot.exists() {
            let count = snapshot.value as? Int ?? 0
            dateSignUpData[slotYourLookingFor] = count //update this array element with the new count from Firebaes
        } else { //didn't find the child node at all, will create the date node and add child nodes with val = 0
            self.REF_CALENDAR_SIGNUPS.child(selectedDate).updateChildValues(dateSignUpData)
        }
        //do something with the dateSignUpData array
    })
}

说明:

传入您感兴趣的日期。代码将读取您要查找的子节点“J9-10A”的节点。

如果存在,它将读取该数据并使用该子节点的计数更新 dateSignUpData 数组。然后您可以进一步处理更新后的数组。

如果“J9-10A”节点不存在,则会创建一个传入日期的节点,然后将所有子节点槽位设置为0。然后您可以进一步处理该数组。

最后一件事。将日期存储为 yyyymmdd 以便订购可能是个好主意。

关于ios - Swift 不检索 Firebase 值(返回值作为 NSNull),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55672181/

相关文章:

ios - swift 中的 Firebase 查询排序顺序?

objective-c - iOS底部标签栏标题

ios - Core Data addPersistentStoreWithType 返回 nil,但错误也是 nil

ios - 强制 UIViewController 正确布局而不将其显示在屏幕上

swift - 在 Apples Swift 3 beta 6 中使用 withMemoryRebound

android - 如何从 Fabric Crashlytics 迁移到 Firebase Crashlytics?

angular - 在加载所有函数/数据之前,不要显示 .html 组件。 Angular 2

ios - 找不到接受提供的参数的 'logInWithPermissions' 的重载

ios - 如何调整添加为 UITableViewController header 的 UIView 的高度

swift - Swift 3 中的 UICollectionView + NSFetchedResultsController