#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;
Комментарии:
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")
}
}
}