#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) }
}
}