Как получить данные Firebase в качестве модели в swift?

#swift #firebase #model #nsdictionary #encodable

#swift #firebase #Модель #nsdictionary #кодируемый

Вопрос:

Я пытаюсь получить данные из firebase и использовать их в качестве модели, которую я создал.

Вот моя модель;

 class UserData{
var nickname : String = ""
var onesignal_player_id : String = ""
var step_count : Int = 0
var total_point : Int = 0
var competitions : [String:Competition] = [String:Competition]()
}

class Competition{
    var end_date : String = ""
    var gift : String = ""
    var id: String = ""
    var name: String = ""
    var users : [String:Int] = [:]
}
 

и это моя функция;

 func getFirebaseData() {
    ref = Database.database().reference()
    ref.child("users").child("HXXNCXf6RRS4WVO12shZ3j15BnG3").observe(.value) { (snapshot) in
        if let snapshotValue = snapshot.value as? Dictionary<String,Any> {
            
            //change userData with the snapshotValue
            self.userData.nickname = snapshotValue["nickname"] as! String
            self.userData.step_count = snapshotValue["step_count"] as! Int
            self.userData.total_point = snapshotValue["total_point"] as! Int
            self.userData.onesignal_player_id = snapshotValue["onesignal_player_id"] as! String
            self.userData.competitions = snapshotValue["competitions"] as! [String:Competition]
            //reload UI with userData

            print(self.userData.competitions)
            
        } else {
            print("An error occured while assigning snapshotValue to userData")
        }
    }
}
 

Этот код выдал мне ошибку, подобную этой;

Не удалось преобразовать значение типа ‘__NSDictionaryM’ (0x1f47ada20) в ‘StepCounterApp.Competition’ (0x1004c06f0). 2021-01-02 23:05:49.985711 0300 StepCounterApp[32511:3685645] Не удалось преобразовать значение типа ‘__NSDictionaryM’ (0x1f47ada20) в ‘StepCounterApp.Competition’ (0x1004c06f0).

но когда я комментирую строку, включенную self.userData.competitions из getFirebaseData функции, все работает нормально.

Что мне делать? Как я могу использовать данные Firebase в качестве модели?

Наконец, вот мои данные firebase;

Данные Firebase

Комментарии:

1. Неясно, что хранится в competitions — это массив Firebase, например, competitions/ 0, competitions / 1, competitions / 2 и т. Д.? Если это так, вы можете переосмыслить эту структуру, поскольку с массивами сложно работать в RTDB, и, вероятно, есть лучшая структура. Кроме того, если вы когда-нибудь захотите обновить какие-либо дочерние данные в пользователях , вам нужно будет включить ключ узла в свои модели Swift, чтобы вы знали, к какому узлу принадлежат данные. Кроме того , почему вы храните конкурсы как [String:Competition] ? В этом нет необходимости, просто массив соревнований был бы лучше.

Ответ №1:

Проблема в вашей модели данных. Объявите свои данные модели следующим образом

 class UserData {
    var nickname : String = ""
    var onesignal_player_id : String = ""
    var step_count : Int = 0
    var total_point : Int = 0
    var competitions : Competition = Competition()
}

class Competition{
    var end_date : String = ""
    var gift : String = ""
    var id: String = ""
    var name: String = ""
    var users : [String:Int] = [:]
    
    init() {
    }
    
    init(with dictionary: [String: Any]) {
        self.end_date = dictionary["end_date"] as! String
        self.gift = dictionary["gift"] as! String
        self.id = dictionary["id"] as! String
        self.name = dictionary["name"] as! String
        self.users = dictionary["users"] as! [String:Int]
    }
}
 

И внутри функции getFirebaseData

 self.userData.competitions = Competition(with: snapshotValue["competitions"] as! [String: Any])
 

Комментарии:

1. Произошла ошибка «Фатальная ошибка: неожиданно найдено nil при развертывании необязательного значения: файл StepCounterApp/UserData.swift, строка 66». строка 66 находится в инициализации со словарем в созданной нами модели.

2. Можете ли вы попробовать распечатать данные snapshotValue[«соревнования»] и проверить тип данных?

3. соревнования var выглядят как массив (это множественное число), поэтому эта модель не будет работать.

4. это результат печати результата snapshotValue[«соревнования»]

Ответ №2:

Проблема была в моей модели данных, и с помощью предложения модели данных Раджи Кишана я исправил проблему.

Сначала я немного изменил модель;

 class UserData{
    var nickname : String = ""
    var onesignal_player_id : String = ""
    var step_count : Int = 0
    var total_point : Int = 0
    var competitions : [String:Competition] = [String:Competition]()
}

class Competition{
    var end_date : String = ""
    var gift : String = ""
    var id: Int = 0
    var name: String = ""
    var users : [String:Int] = [:]

    init() {
    }

    init(with dictionary: [String: Any]) {
        self.end_date = dictionary["end_date"] as! String
        self.gift = dictionary["gift"] as! String
        self.id = dictionary["id"] as! Int
        self.name = dictionary["name"] as! String
        self.users = dictionary["users"] as! [String:Int]
    }
}
 

Затем я добавляю childSnapshot в свой метод, чтобы я мог напрямую работать с «соревнованиями»;

 func getFirebaseData() {
    ref = Database.database().reference()
    ref.child("users").child("HXXNCXf6RRS4WVO12shZ3j15BnG3").observe(.value) { [self] (snapshot) in
        if let snapshotValue = snapshot.value as? [String:Any] {
            
            //change userData with the snapshotValue
            self.userData.nickname = snapshotValue["nickname"] as! String
            self.userData.step_count = snapshotValue["step_count"] as! Int
            self.userData.total_point = snapshotValue["total_point"] as! Int
            self.userData.onesignal_player_id = snapshotValue["onesignal_player_id"] as! String
            
            //******
            //This part of the coded added for to solve the problem starting from here 
            let childSnap = snapshot.childSnapshot(forPath: "competitions")
            if let childSnapValue = childSnap.value as? [String:Any] {
                childSnapValue.forEach { (element) in
                    self.userData.competitions.updateValue(Competition(with: element.value as! [String:Any]), forKey: element.key)
                }
            } else {
                print("something wrong with the childSnap")
            }
            //to here
            //******

        } else {
            print("An error occured while assigning snapshotValue to userData")
        }
    }
}