На самом деле ответы выше действительно хороши, но в них отсутствуют некоторые детали того, что нужно многим людям в постоянно развивающемся проекте клиент / сервер. Мы разрабатываем приложение, в то время как наш бэкэнд постоянно развивается с течением времени, что означает, что некоторые перечисленные случаи изменят эту эволюцию. Поэтому нам нужна стратегия декодирования перечислений, которая способна декодировать массивы перечислений, которые содержат неизвестные регистры. В противном случае декодирование объекта, содержащего массив, просто не удастся.
То, что я сделал, довольно просто:
enum Direction: String, Decodable {
case north, south, east, west
}
struct DirectionList {
let directions: [Direction]
}
extension DirectionList: Decodable {
public init(from decoder: Decoder) throws {
var container = try decoder.unkeyedContainer()
var directions: [Direction] = []
while !container.isAtEnd {
// Here we just decode the string from the JSON which always works as long as the array element is a string
let rawValue = try container.decode(String.self)
guard let direction = Direction(rawValue: rawValue) else {
// Unknown enum value found - ignore, print error to console or log error to analytics service so you'll always know that there are apps out which cannot decode enum cases!
continue
}
// Add all known enum cases to the list of directions
directions.append(direction)
}
self.directions = directions
}
}
Бонус: Скрыть реализацию> Сделать коллекцию
Скрывать детали реализации всегда хорошая идея. Для этого вам понадобится немного больше кода. Хитрость заключается в том, чтобы соответствовать , DirectionsList
чтобы Collection
и сделать свой внутренний list
массив частным:
struct DirectionList {
typealias ArrayType = [Direction]
private let directions: ArrayType
}
extension DirectionList: Collection {
typealias Index = ArrayType.Index
typealias Element = ArrayType.Element
// The upper and lower bounds of the collection, used in iterations
var startIndex: Index { return directions.startIndex }
var endIndex: Index { return directions.endIndex }
// Required subscript, based on a dictionary index
subscript(index: Index) -> Element {
get { return directions[index] }
}
// Method that returns the next index when iterating
func index(after i: Index) -> Index {
return directions.index(after: i)
}
}
Вы можете прочитать больше о соответствии пользовательских коллекций в этом сообщении в блоге Джона Санделла: https://medium.com/@johnsundell/creating-custom-collections-in-swift-a344e25d0bb0