如何在Firebase中保存嵌套数据模型? Swift错误:只能存储>的对象输入NSNumber,NSString,NSDictionary和NSArray

时间:2017-12-13 13:26:04

标签: ios arrays firebase firebase-realtime-database anyobject

当我尝试在Firebase中保存嵌套模型时,我收到下面列出的错误。这是因为数据模型格式不正确,但我搜索了几个小时,现在我被卡住了。

  

无法存储_SwiftValue类型的对象。只能存储对象   输入NSNumber,NSString,NSDictionary和NSArray。

我想写的模型如下:

payPeriodDateStartDate
payPeriodEndDate
  -bookings 
     booking1
          hours: 
          originalPriceOfBooking:
          cancelledBy   // <- optional, it may or may not exist
             -timeStamp // <- a timeStamp node
                cleanerUID:
                costToCancelByCleaner:

     booking2 

我认为我没有正确转换此代码段中的cancelledBy

       for booking in self.bookingsFromQuery { 
          let bookingForPayPeriod = DisbursePaymentData(
           numberOfHours: booking.NumberOfHours
            cancelledBy: booking.objectsUnderCancelledBy) // <- this argument is not properly converted to AnyObject and causes exception 'InvalidFirebaseData
        )

您可以在下面找到完整的代码。

 //FireBaseData is used to read from Firebase
struct FireBaseData {

  var BookingNumber:String!
  var PostCode:String!
  var objectsUnderCancelledBy:[String: CancelledObject]! //the object that is causing the crash of the app, it is optional, cleaner may/may not cancel booking

    init(snapshot:FIRDataSnapshot){
    //I am initializing just the property we are concerned about, in my project all properties are initialized 

    //CancelledBy is nested
  if ((snapshot.value! as? NSDictionary)?["BookingNumber"] as? String) != nil {    
        var cancelledItems = [String: CancelledObject]()

        //each key:value under the bookingNumber
        for item in snapshot.children{
          let elementUnderBookingNumber = item as! FIRDataSnapshot
            if elementUnderBookingNumber.key == "CancelledBy" {

                //iterate through the elements under each timeStamp
                for objectUnderTimeStamp in elementUnderBookingNumber.children {
                    let item = CancelledObject(snapshot: objectUnderTimeStamp as! FIRDataSnapshot)
                    //assign the timeStamp as a key for retrieved item
                    cancelledItems["\(item.key!)"] = item
                }
            }
        }
          self.objectsUnderCancelledBy = cancelledItems
   }//end of init
}//end of struct FirebaseData



  //holds data under CancelledBy key when data is read from Firebase 
 class CancelledObject {

     var cleanerUID: String!
     var costToCancelByCleaner: String!
    init() {
           if let cleanerUIDContent = (snapshot.value! as? NSDictionary)?["CleanerUID"] as? String {
         self.cleanerUID = cleanerUIDContent
       }
    }
 }//end of CancelledObject


     //DisbursePaymentData is used to update the database
     struct DisbursePaymentData {

     var payPeriodDateStartDate:String!
     var payPeriodEndDate:String!
     //- optional, cancelledBy might not exist in Database
     var cancelledBy: [String: CancelledObject]! 
     var numberOfHours:String!
       init(
    payPeriodDateStartDate:String,
    payPeriodEndDate:String) {
       self.payPeriodDateStartDate = payPeriodDateStartDate
       self.payPeriodEndDate = payPeriodEndDate
     }

    init(
      numberOfHours:String,
      cancelledBy: [String:CancelledObject]!
     ) {
        self.numberOfHours = numberOfHours
        self.cancelledBy =  cancelledBy
     } 


   //enables to convert the values to AnyObject
func convertDisbursePaymentDataToAnyObject() -> AnyObject {
    var someDict = [String : AnyObject]()

    someDict["numberOfHours"] = self.numberOfHours as AnyObject?
    someDict["cancelledBy"] = self.cancelledBy as AnyObject? //<- is this the way to convert cancelledBy to AnyObject?
    return someDict as AnyObject
}


}//end of struct DisbursePaymentData





  class UpdateFirebase {

      var bookingsFromQuery = [FireBaseData]()

       override func viewDidLoad() {
         super.viewDidLoad()
        //data was successfully read from Firebase

       var finalBookingArray = [String: AnyObject]()

      for booking in self.bookingsFromQuery { 
          let bookingForPayPeriod = DisbursePaymentData(

           numberOfHours: booking.NumberOfHours
          cancelledBy: booking.objectsUnderCancelledBy) // <- this argument is not properly converted to AnyObject and causes exception 'InvalidFirebaseData
        )
           //convert each booking to AnyObject
          let bookingCompleted = bookingForPayPeriod.convertDisbursePaymentDataToAnyObject()
   //assign to BookingNumber of each loop the converted object
         finalBookingArray[booking.BookingNumber] = bookingCompleted
    }//end of for loop

     let finalObject = DisbursePaymentData(
            payPeriodDateStartDate: self.fromDateString.text!,
            payPeriodEndDate: self.toDateString.text!)

      let childUpdates = ["Payments/\(uid)/\(timeStamp)/\(paymentRef)" : finalObject,
                          "Payments/\(uid)/\(timeStamp)/\(paymentRef)/\("bookings")" : finalBookingArray] as [String : Any] 

       self.dbRef.updateChildValues(childUpdates)
   }//end of viewDidLoad
 }

0 个答案:

没有答案