Swift — кодируемый массив декодирования массивов CLLocation

#swift #codable #cllocation #decodable

#swift #кодируемый #cllocation #декодируемый

Вопрос:

У меня есть структура, которая содержит массив массивов CLLocations. Это делается для поддержки многополилинии (другими словами, набора потенциально несмежных строк). Я хочу кодировать и декодировать эти данные. У меня возникли проблемы с написанием методов кодирования и декодирования, поскольку CLLocation по умолчанию не кодируется.

     struct MyTrack {
      let coords: [[CLLocation]]?
    
      enum CodingKeys: String, CodingKey {
        case coords
      }
    }

    extension MyTrack: Decodable {
      init(from decoder: Decoder) throws {
        let values = try decoder.container(keyedBy: CodingKeys.self)
        
        coords = try values.decodeIfPresent([[CLLocation]].self, forKey: .coords)?
            .map { ($0 as AnyObject).map { CLLocation(model: $0) } }
    }
}
  

В настоящее время он выдает две ошибки в Xcode:

Не удается преобразовать значение типа ‘[[CLLocation]].Введите ‘ожидаемый тип аргумента ‘[Любой?].Введите’

Значение типа ‘AnyObject’ не имеет элемента ‘map’

Любая помощь очень ценится!



Поскольку CLLocation по умолчанию не поддается кодированию, я следовал руководству по созданию структуры-оболочки вокруг него, код выглядит следующим образом:

 extension CLLocation: Encodable {
    enum CodingKeys: String, CodingKey {
        case latitude
        case longitude
        case altitude
        case horizontalAccuracy
        case verticalAccuracy
        case speed
        case course
        case timestamp
    }
    public func encode(to encoder: Encoder) throws {
        var container = encoder.container(keyedBy: CodingKeys.self)
        try container.encode(coordinate.latitude, forKey: .latitude)
        try container.encode(coordinate.longitude, forKey: .longitude)
        try container.encode(altitude, forKey: .altitude)
        try container.encode(horizontalAccuracy, forKey: .horizontalAccuracy)
        try container.encode(verticalAccuracy, forKey: .verticalAccuracy)
        try container.encode(speed, forKey: .speed)
        try container.encode(course, forKey: .course)
        try container.encode(timestamp, forKey: .timestamp)
    }
}

struct Location: Codable {
    let latitude: CLLocationDegrees
    let longitude: CLLocationDegrees
    let altitude: CLLocationDistance
    let horizontalAccuracy: CLLocationAccuracy
    let verticalAccuracy: CLLocationAccuracy
    let speed: CLLocationSpeed
    let course: CLLocationDirection
    let timestamp: Date
}

extension CLLocation {
    convenience init(model: Location) {
      self.init(coordinate: CLLocationCoordinate2DMake(model.latitude, model.longitude), altitude: model.altitude, horizontalAccuracy: model.horizontalAccuracy, verticalAccuracy: model.verticalAccuracy, course: model.course, speed: model.speed, timestamp: model.timestamp)
     }
}
  

Ответ №1:

Вы декодируете CLLocation , а не свою структуру-оболочку. Вместо этого вы должны декодировать свою структуру-оболочку. Кроме того, вы не должны приводить к AnyObject .

 extension MyTrack: Decodable {
    init(from decoder: Decoder) throws {
        let values = try decoder.container(keyedBy: CodingKeys.self)
    
        coords = try values.decodeIfPresent([[Location]].self, forKey: .coords)?
            .map { $0.map(CLLocation.init) }
    }
}