Декодирование вложенного JSON с помощью Swift 4 возвращает только один объект
Я пытаюсь декодировать JSON, полученный от сетевого запроса, но он декодирует только первый объект. JSON выглядит так
{
"lastUpdatedOn": "2018-08-21T14:38:38.251Z",
"players": [
{
"player": {
"id": 10138,
"firstName": "Alex",
"lastName": "Abrines",
"primaryPosition": "SG",
"alternatePositions": [],
"jerseyNumber": 8,
"currentTeam": {
"id": 96,
"abbreviation": "OKL"
},
"currentRosterStatus": "ROSTER",
"currentInjury": null,
"height": "6'6\"",
"weight": 190,
"birthDate": "1993-08-01",
"age": 25,
"birthCity": "Palma de Mallorca",
"birthCountry": "Spain",
"rookie": false,
"highSchool": null,
"college": null,
"handedness": {
"shoots": "R"
},
"officialImageSrc": "https://ak-static.cms.nba.com/wp-content/uploads/headshots/nba/latest/260x190/203518.png",
"socialMediaAccounts": [],
"currentContractYear": null,
"drafted": {
"year": 2013,
"team": {
"id": 96,
"abbreviation": "OKL"
},
"pickTeam": {
"id": 96,
"abbreviation": "OKL"
},
"round": 2,
"roundPick": 2,
"overallPick": 32
},
"externalMappings": [
{
"source": "NBA.com",
"id": 203518
}
]
},
"teamAsOfDate": {
"id": 96,
"abbreviation": "OKL"
}
},
{
"player": {
"id": 9466,
"firstName": "Quincy",
"lastName": "Acy",
"primaryPosition": "PF",
"alternatePositions": [],
"jerseyNumber": 13,
"currentTeam": {
"id": 84,
"abbreviation": "BRO"
},
"currentRosterStatus": "ROSTER",
"currentInjury": null,
"height": "6'7\"",
"weight": 240,
"birthDate": "1990-10-06",
"age": 27,
"birthCity": "Tyler, TX",
"birthCountry": "USA",
"rookie": false,
"highSchool": "Mesquite Horn",
"college": "Baylor University",
"handedness": {
"shoots": "R"
},
"officialImageSrc": "https://ak-static.cms.nba.com/wp-content/uploads/headshots/nba/latest/260x190/203112.png",
"socialMediaAccounts": [
{
"mediaType": "TWITTER",
"value": "QuincyAcy"
}
],
"currentContractYear": {
"seasonStartYear": 2017,
"baseSalary": 1709538,
"minorsSalary": 0,
"signingBonus": 0,
"otherBonuses": 0,
"capHit": 1709538,
"fullNoTradeClause": false,
"modifiedNoTradeClause": false,
"noMovementClause": false,
"overallContract": {
"signingTeam": {
"id": 84,
"abbreviation": "BRO"
},
"signedOn": null,
"totalYears": 2,
"totalSalary": 3499630,
"totalBonuses": 0,
"expiryStatus": "UFA",
"annualAverageSalary": 1749815
}
},
"drafted": {
"year": 2012,
"team": {
"id": 81,
"abbreviation": "TOR"
},
"pickTeam": {
"id": 81,
"abbreviation": "TOR"
},
"round": 2,
"roundPick": 7,
"overallPick": 37
},
"externalMappings": [
{
"source": "NBA.com",
"id": 203112
}
]
},
"teamAsOfDate": {
"id": 84,
"abbreviation": "BRO"
}
},
{
"player": {
"id": 9301,
"firstName": "Jordan",
"lastName": "Adams",
"primaryPosition": "SG",
"alternatePositions": [],
"jerseyNumber": 3,
"currentTeam": null,
"currentRosterStatus": "RETIRED",
"currentInjury": null,
"height": "6'5\"",
"weight": 209,
"birthDate": "1994-07-08",
"age": 24,
"birthCity": "Atlanta, GA",
"birthCountry": "USA",
"rookie": false,
"highSchool": "Oak Hill Academy",
"college": "University of California, Los Angeles",
"handedness": {
"shoots": "R"
},
"officialImageSrc": "https://ak-static.cms.nba.com/wp-content/uploads/headshots/nba/latest/260x190/203919.png",
"socialMediaAccounts": [
{
"mediaType": "TWITTER",
"value": "jordanadams1231"
}
],
"currentContractYear": null,
"drafted": {
"year": 2014,
"team": {
"id": 107,
"abbreviation": "MEM"
},
"pickTeam": {
"id": 107,
"abbreviation": "MEM"
},
"round": 1,
"roundPick": 22,
"overallPick": 22
},
"externalMappings": [
{
"source": "NBA.com",
"id": 203919
}
]
},
"teamAsOfDate": null
}
],
"references": {
"teamReferences": [
{
"id": 84,
"city": "Brooklyn",
"name": "Nets",
"abbreviation": "BRO",
"homeVenue": {
"id": 78,
"name": "Barclays Center"
},
"teamColoursHex": [],
"socialMediaAccounts": [],
"officialLogoImageSrc": null
},
{
"id": 96,
"city": "Oklahoma City",
"name": "Thunder",
"abbreviation": "OKL",
"homeVenue": {
"id": 90,
"name": "Chesapeake Energy Arena"
},
"teamColoursHex": [],
"socialMediaAccounts": [],
"officialLogoImageSrc": null
}
]
}
}
Моя структура в настоящее время выглядит следующим образом, и я пытаюсь использовать пользовательский декодер, я смог заставить его работать, используя вложенные структуры, но затем мне нужно перебрать декодированные данные и добавить все проигрыватели в массив.
struct Player: Codable {
let firstName: String
let lastName: String
enum RootKeys: String, CodingKey {
case players
}
enum PlayerKeys: String, CodingKey {
case player
}
init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: RootKeys.self)
var playersUnkeyedContainer = try container.nestedUnkeyedContainer(forKey: .players)
let playersContainer = try playersUnkeyedContainer.nestedContainer(keyedBy: PlayerKeys.self)
let playerContainer = try playersContainer.nestedContainer(keyedBy: Player.CodingKeys.self, forKey: .player)
firstName = try playerContainer.decode(String.self, forKey: .firstName)
lastName = try playerContainer.decode(String.self, forKey: .lastName)
}
}
Когда я пытаюсь сделать это
let decodedPlayers = try jsonDecoder.decode(Player.self, from: data)
Это работает, но decodedPlayers - только один Player, и если я попытаюсь сделать это:
let decodedPlayers = try jsonDecoder.decode([Player].self, from: data)
Я получаю это сообщение об ошибке: "Предполагается декодировать массив, но вместо этого нашел словарь".
4 ответа
Это работает:
struct Player: Decodable {
let firstName: String
let lastName: String
}
struct AllPlayers: Decodable {
let players: [NBAPlayer]
}
struct NBAPlayer: Decodable {
let player: Player
}
Но тогда, если я хочу массив всех объектов Player, я должен сделать это:
var players = [Player]()
do {
let decodedPlayers = try jsonDecoder.decode(AllPlayers.self, from: data)
for player in decodedPlayers.players {
players.append(player.player)
}
print(players)
} catch {
NSLog("Error decoding players :\(error)")
}
Это помогает тебе!!
// To parse the JSON, add this file to your project and do:
//
// let welcome = try? newJSONDecoder().decode(Welcome.self, from: jsonData)
import Foundation
struct Welcome: Codable {
let lastUpdatedOn: String
let players: [PlayerElement]
let references: References
}
struct PlayerElement: Codable {
let player: PlayerPlayer
let teamAsOfDate: TeamAsOfDate?
}
struct PlayerPlayer: Codable {
let id: Int
let firstName, lastName, primaryPosition: String
let alternatePositions: [JSONAny]
let jerseyNumber: Int
let currentTeam: TeamAsOfDate?
let currentRosterStatus: String
let currentInjury: JSONNull?
let height: String
let weight: Int
let birthDate: String
let age: Int
let birthCity, birthCountry: String
let rookie: Bool
let highSchool, college: String?
let handedness: Handedness
let officialImageSrc: String
let socialMediaAccounts: [SocialMediaAccount]
let currentContractYear: CurrentContractYear?
let drafted: Drafted
let externalMappings: [ExternalMapping]
}
struct CurrentContractYear: Codable {
let seasonStartYear, baseSalary, minorsSalary, signingBonus: Int
let otherBonuses, capHit: Int
let fullNoTradeClause, modifiedNoTradeClause, noMovementClause: Bool
let overallContract: OverallContract
}
struct OverallContract: Codable {
let signingTeam: TeamAsOfDate
let signedOn: JSONNull?
let totalYears, totalSalary, totalBonuses: Int
let expiryStatus: String
let annualAverageSalary: Int
}
struct TeamAsOfDate: Codable {
let id: Int
let abbreviation: String
}
struct Drafted: Codable {
let year: Int
let team, pickTeam: TeamAsOfDate
let round, roundPick, overallPick: Int
}
struct ExternalMapping: Codable {
let source: String
let id: Int
}
struct Handedness: Codable {
let shoots: String
}
struct SocialMediaAccount: Codable {
let mediaType, value: String
}
struct References: Codable {
let teamReferences: [TeamReference]
}
struct TeamReference: Codable {
let id: Int
let city, name, abbreviation: String
let homeVenue: HomeVenue
let teamColoursHex, socialMediaAccounts: [JSONAny]
let officialLogoImageSrc: JSONNull?
}
struct HomeVenue: Codable {
let id: Int
let name: String
}
// MARK: Encode/decode helpers
class JSONNull: Codable, Hashable {
public static func == (lhs: JSONNull, rhs: JSONNull) -> Bool {
return true
}
public var hashValue: Int {
return 0
}
public init() {}
public required init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
if !container.decodeNil() {
throw DecodingError.typeMismatch(JSONNull.self, DecodingError.Context(codingPath: decoder.codingPath, debugDescription: "Wrong type for JSONNull"))
}
}
public func encode(to encoder: Encoder) throws {
var container = encoder.singleValueContainer()
try container.encodeNil()
}
}
class JSONCodingKey: CodingKey {
let key: String
required init?(intValue: Int) {
return nil
}
required init?(stringValue: String) {
key = stringValue
}
var intValue: Int? {
return nil
}
var stringValue: String {
return key
}
}
class JSONAny: Codable {
let value: Any
static func decodingError(forCodingPath codingPath: [CodingKey]) -> DecodingError {
let context = DecodingError.Context(codingPath: codingPath, debugDescription: "Cannot decode JSONAny")
return DecodingError.typeMismatch(JSONAny.self, context)
}
static func encodingError(forValue value: Any, codingPath: [CodingKey]) -> EncodingError {
let context = EncodingError.Context(codingPath: codingPath, debugDescription: "Cannot encode JSONAny")
return EncodingError.invalidValue(value, context)
}
static func decode(from container: SingleValueDecodingContainer) throws -> Any {
if let value = try? container.decode(Bool.self) {
return value
}
if let value = try? container.decode(Int64.self) {
return value
}
if let value = try? container.decode(Double.self) {
return value
}
if let value = try? container.decode(String.self) {
return value
}
if container.decodeNil() {
return JSONNull()
}
throw decodingError(forCodingPath: container.codingPath)
}
static func decode(from container: inout UnkeyedDecodingContainer) throws -> Any {
if let value = try? container.decode(Bool.self) {
return value
}
if let value = try? container.decode(Int64.self) {
return value
}
if let value = try? container.decode(Double.self) {
return value
}
if let value = try? container.decode(String.self) {
return value
}
if let value = try? container.decodeNil() {
if value {
return JSONNull()
}
}
if var container = try? container.nestedUnkeyedContainer() {
return try decodeArray(from: &container)
}
if var container = try? container.nestedContainer(keyedBy: JSONCodingKey.self) {
return try decodeDictionary(from: &container)
}
throw decodingError(forCodingPath: container.codingPath)
}
static func decode(from container: inout KeyedDecodingContainer<JSONCodingKey>, forKey key: JSONCodingKey) throws -> Any {
if let value = try? container.decode(Bool.self, forKey: key) {
return value
}
if let value = try? container.decode(Int64.self, forKey: key) {
return value
}
if let value = try? container.decode(Double.self, forKey: key) {
return value
}
if let value = try? container.decode(String.self, forKey: key) {
return value
}
if let value = try? container.decodeNil(forKey: key) {
if value {
return JSONNull()
}
}
if var container = try? container.nestedUnkeyedContainer(forKey: key) {
return try decodeArray(from: &container)
}
if var container = try? container.nestedContainer(keyedBy: JSONCodingKey.self, forKey: key) {
return try decodeDictionary(from: &container)
}
throw decodingError(forCodingPath: container.codingPath)
}
static func decodeArray(from container: inout UnkeyedDecodingContainer) throws -> [Any] {
var arr: [Any] = []
while !container.isAtEnd {
let value = try decode(from: &container)
arr.append(value)
}
return arr
}
static func decodeDictionary(from container: inout KeyedDecodingContainer<JSONCodingKey>) throws -> [String: Any] {
var dict = [String: Any]()
for key in container.allKeys {
let value = try decode(from: &container, forKey: key)
dict[key.stringValue] = value
}
return dict
}
static func encode(to container: inout UnkeyedEncodingContainer, array: [Any]) throws {
for value in array {
if let value = value as? Bool {
try container.encode(value)
} else if let value = value as? Int64 {
try container.encode(value)
} else if let value = value as? Double {
try container.encode(value)
} else if let value = value as? String {
try container.encode(value)
} else if value is JSONNull {
try container.encodeNil()
} else if let value = value as? [Any] {
var container = container.nestedUnkeyedContainer()
try encode(to: &container, array: value)
} else if let value = value as? [String: Any] {
var container = container.nestedContainer(keyedBy: JSONCodingKey.self)
try encode(to: &container, dictionary: value)
} else {
throw encodingError(forValue: value, codingPath: container.codingPath)
}
}
}
static func encode(to container: inout KeyedEncodingContainer<JSONCodingKey>, dictionary: [String: Any]) throws {
for (key, value) in dictionary {
let key = JSONCodingKey(stringValue: key)!
if let value = value as? Bool {
try container.encode(value, forKey: key)
} else if let value = value as? Int64 {
try container.encode(value, forKey: key)
} else if let value = value as? Double {
try container.encode(value, forKey: key)
} else if let value = value as? String {
try container.encode(value, forKey: key)
} else if value is JSONNull {
try container.encodeNil(forKey: key)
} else if let value = value as? [Any] {
var container = container.nestedUnkeyedContainer(forKey: key)
try encode(to: &container, array: value)
} else if let value = value as? [String: Any] {
var container = container.nestedContainer(keyedBy: JSONCodingKey.self, forKey: key)
try encode(to: &container, dictionary: value)
} else {
throw encodingError(forValue: value, codingPath: container.codingPath)
}
}
}
static func encode(to container: inout SingleValueEncodingContainer, value: Any) throws {
if let value = value as? Bool {
try container.encode(value)
} else if let value = value as? Int64 {
try container.encode(value)
} else if let value = value as? Double {
try container.encode(value)
} else if let value = value as? String {
try container.encode(value)
} else if value is JSONNull {
try container.encodeNil()
} else {
throw encodingError(forValue: value, codingPath: container.codingPath)
}
}
public required init(from decoder: Decoder) throws {
if var arrayContainer = try? decoder.unkeyedContainer() {
self.value = try JSONAny.decodeArray(from: &arrayContainer)
} else if var container = try? decoder.container(keyedBy: JSONCodingKey.self) {
self.value = try JSONAny.decodeDictionary(from: &container)
} else {
let container = try decoder.singleValueContainer()
self.value = try JSONAny.decode(from: container)
}
}
public func encode(to encoder: Encoder) throws {
if let arr = self.value as? [Any] {
var container = encoder.unkeyedContainer()
try JSONAny.encode(to: &container, array: arr)
} else if let dict = self.value as? [String: Any] {
var container = encoder.container(keyedBy: JSONCodingKey.self)
try JSONAny.encode(to: &container, dictionary: dict)
} else {
var container = encoder.singleValueContainer()
try JSONAny.encode(to: &container, value: self.value)
}
}
}
Решение с вложенными контейнерами кажется довольно громоздким
Мое предложение состоит в том, чтобы декодировать структуры традиционным способом с помощью вложенных структур.
struct Root: Decodable {
let lastUpdatedOn : String
let players : [PlayerData]
}
struct PlayerData: Decodable {
let player : Player
let teamAsOfDate : TeamAsOfDate?
}
struct TeamAsOfDate: Decodable {
let id : Int
let abbreviation : String
}
struct Player: Decodable {
let firstName, lastName : String
}
do {
let result = try JSONDecoder().decode(Root.self, from: data)
print(result)
} catch { print(error) }
Ваш JSON не является массивом. Таким образом, вы не можете использовать:
let decodedPlayers = try jsonDecoder.decode([Player].self, from: data)
Тебе нужно:
[{"id": 10138, "firstName": "Alex"}, {"id": 10139, "firstName": "Alex2"} ]
ИЛИ ЖЕ
struct YourData: Codable {
let lastUpdatedOn: String
let players: [Player]
}
extension YourData: Decodable {
enum CodingKeys: CodingKey { case lastUpdatedOn, players }
enum NodeKeys: CodingKey { case nodes }
init(from decoder: Decoder) throws {
let rootContainer = try decoder.container(keyedBy: CodingKeys.self)
try lastUpdatedOn = rootContainer.decode(String.self, forKey: .lastUpdatedOn)
let playersNode = try rootContainer.nestedContainer(keyedBy: NodeKeys.self, forKey: .players)
var nodes = try playersNode(forKey: .nodes)
var playerArray: [Player] = []
while !nodes.isAtEnd {
let ply = try nodes.decode(Player.self)
playerArray += [ply]
}
players = playerArray
}
}
В заключение:
let decodedPlayers = try jsonDecoder.decode(YourData.self, from: data)
Рекомендации: