草庐IT

ios - 使用 Swift 4 解码嵌套 JSON 只返回一个对象

coder 2024-01-28 原文

我正在尝试解码从网络请求中收到的 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 只是一个播放器,如果我尝试这样做:

let decodedPlayers = try jsonDecoder.decode([Player].self, from: data)

我收到此错误消息: “本应解码数组,但却发现了一本字典。”

最佳答案

这对你有帮助!!

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

关于ios - 使用 Swift 4 解码嵌套 JSON 只返回一个对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53941204/

有关ios - 使用 Swift 4 解码嵌套 JSON 只返回一个对象的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  4. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  5. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  6. ruby - 在 Ruby 中使用匿名模块 - 2

    假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

  7. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  8. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  9. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  10. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

随机推荐