【问题标题】:Swift Json decoding nested array / dictionary to flat modelSwift Json 将嵌套数组/字典解码为平面模型
【发布时间】:2021-05-03 19:11:38
【问题描述】:

我正在尝试将以下 json 对象解码为我在 Swift 中的 User 模型。

我的问题是解码 tokens 数组中的值 _idtoken,其中数组中的第一个标记包含我要解码为 User.tokenId 和 User.token 的值。

我正在尝试将值直接提取/映射到我的用户模型结构中,而我的用户模型中没有另一个嵌套结构(例如struct Token { var id: String , var token: String }

let json = """
    {
        "currentLocation": {
            "latitude": 0,
            "longitude": 0
        },
        "profileImageUrl": "",
        "bio": "",
        "_id": "601453e4aae564fc19075b68",
        "username": "johnsmith",
        "name": "john",
        "email": "johnsmith@gmail.com",
        "keywords": ["word", "weds"],
        "tokens": [
            {
                "_id": "213453e4aae564fcqu775b69",
                "token": "eyJhbGciOiJIUzqoNiIsInR5cCI6IkpXVCJ9.eyJfaWQiOiI2MDE0NTNlNGFhZTU2NGZjMTkwNzViNjgiLCJpYXQiOjE2MTE5NDQ5MzIsImV4cCI6MTYxMjM3NjkzMn0.PbTsA3B0MAfcVvEF1UAMhUXFiqIL1FcxVFGgMTZ5HCk"
            }
        ],
        "createdAt": "2021-01-29T18:28:52.845Z",
        "updatedAt": "2021-01-29T18:28:52.883Z"
    }
    """.data(using: .utf8)!


struct User: Codable {
    var latitude: Double 
    var longitude: Double 
    var profileImageUrl: String 
    var bio: String 
    var userId: String 
    var username: String
    var name: String
    var email: String
    var keywords: [String]
    var tokenId: String
    var token: String
    var createdAt: Date
    var updatedAt: Date
    
    private enum UserKeys: String, CodingKey {
        case currentLocation
        case profileImageUrl
        case bio
        case userId = "_id"
        case username
        case name
        case email
        case keywords
        case tokens
        case createdAt
        case updatedAt
    }
    
    private enum CurrentLocationKeys: String, CodingKey { 
        case latitude
        case longitude
    }
    
    private enum TokenKeys: String, CodingKey {
        case tokenId = "_id"
        case token
    }
    
    init(from decoder: Decoder) throws {
        
        let userContainer = try decoder.container(keyedBy: UserKeys.self)
              let currentLocationContainer = try userContainer.nestedContainer(keyedBy: CurrentLocationKeys.self, forKey: .currentLocation) 
              self.latitude = try currentLocationContainer.decode(Double.self, forKey: .latitude)
              self.longitude = try currentLocationContainer.decode(Double.self, forKey: .longitude)
            self.profileImageUrl = try userContainer.decode(String.self, forKey: .profileImageUrl)
            self.bio = try userContainer.decode(String.self, forKey: .bio)
            self.userId = try userContainer.decode(String.self, forKey: .userId)
            self.username = try userContainer.decode(String.self, forKey: .username)
            self.name = try userContainer.decode(String.self, forKey: .name)
            self.email = try userContainer.decode(String.self, forKey: .email)
            self.keywords = try userContainer.decode([String].self, forKey: .keywords)
              let tokensContainer = try userContainer.nestedContainer(keyedBy: TokenKeys.self, forKey: .tokens)
              self.tokenId = try tokensContainer.decode(String.self, forKey: .tokenId)
              self.token = try tokensContainer.decode(String.self, forKey: .token)
            self.createdAt = try userContainer.decode(Date.self, forKey: .createdAt)
            self.updatedAt = try userContainer.decode(Date.self, forKey: .updatedAt)
    }
}

let user = try! decoder.decode(User.self, from: json)

【问题讨论】:

  • 您可以编写自定义解码代码,或者(这就是我会做的)使用嵌套结构并在外部结构中提供计算属性来公开嵌套值。
  • 显示错误,但是tokens是一个数组(如果你读它就是控制台的错误),所以如果有多个值,你保留哪一个?
  • 如果某个东西是一个数组,那通常是有原因的。您将需要一种策略来处理 0 个或多个令牌,并且最好在解码对象之外执行此操作。我强烈建议“直接”映射对象并处理解码之外的特殊情况。这将大大减少所需的代码以及认知复杂性。

标签: ios json swift decode datamodel


【解决方案1】:

首先,我假设您的 decoder 具有适当的日期解码策略,能够将 ISO8601 字符串解码为 Date

token 字典的封闭容器是一个数组。你必须插入一个中间nestedUnkeyedContainer

...
var arrayContainer = try userContainer.nestedUnkeyedContainer(forKey: .tokens)
let tokensContainer = try arrayContainer.nestedContainer(keyedBy: TokenKeys.self)
self.tokenId = try tokensContainer.decode(String.self, forKey: .tokenId)
self.token = try tokensContainer.decode(String.self, forKey: .token)
...

将 JSON 解码为多个结构的代码要少得多

【讨论】:

  • 如前所述,避免两个嵌套属性的代码太多。用户可以使用lazy var 将该事实隐藏到更高级别的方法中。
  • 谢谢瓦迪安!是的,我有一个未包含在此代码中的自定义日期解码策略:
猜你喜欢
  • 2020-11-29
  • 1970-01-01
  • 1970-01-01
  • 2022-01-24
  • 2019-11-29
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-08-25
相关资源
最近更新 更多