Swift 原生Json和Model转换

- 4 mins

Apple 在 Swift 4 的 Foundation 的模块中添加了对 JSON 解析成Model的原生支持。

虽然之前也有ObjectMapperSwiftyJSON这样的解决方案,但是Swift 4中添加了原生支持还是不免让人兴奋,使用起来也相对方便了许多。

简单使用

如果你的 JSON 数据结构和你使用的 Model 对象结构一致的话,那么解析过程将会非常简单。只需要让Model声明 Codable 协议即可。

JSON数据:

{
    "userInfos": [
        {
            "userName": "小名",
            "age": 18,
            "height": 178.56,
            "sex": true
        },
        {
            "userName": "小方",
            "age": 18,
            "height": 162.56,
            "sex": false
        }
    ]
}

Model对象:


struct UserList: Codable {

    let userInfos: [UserInfo]

    struct UserInfo: Codable {

        let userName: String
        let age: Int
        let height: Float
        let sex: Bool
    }
}

JSON转Model:

    let jsonDecoder = JSONDecoder()
    let modelObject = try? jsonDecoder.decode(UserList.self, from: jsonData)

Model转JSON:

    let jsonEncoder = JSONEncoder()
    let jsonData = try? jsonEncoder.encode(modelObject)

关于 Codable

/// A type that can convert itself into and out of an external representation.
public typealias Codable = Decodable & Encodable

Codable 同时声明了 DecodableEncodable 两个协议,只需要单向处理的话可以只声明其中一个协议。

需要注意的是,使用时必须保证 JSON 数据结构和你使用的 Model 对象结构一致,任何不一致都会导致解析失败。

但是实际开发过程中,往往不会总是这样满足我们的需要,有时也需要我们对解析过程做一些改变。

自定义键值名

为了保持风格一致,我们有时候不得不改变key成为我们需要的key值。

仍然使用上面的例子,现在我们把Model中的key height 改为 bodyHeight

struct UserList: Codable {

    let userInfos: [UserInfo]
    
    struct UserInfo: Codable {
        
        let userName: String
        let age: Int
        let bodyHeight: Float
        let sex: Bool
        
        private enum CodingKeys: String, CodingKey {
            case userName
            case age
            case bodyHeight = "height"
            case sex
        }
    }
}

格式处理

JSONEncoderJSONDecoder 本身也提供了对时间格式的处理,Data的处理,浮点不匹配的处理。可以根据自己的需要设置自己想要的格式。

    /// The strategy to use in decoding dates. Defaults to `.deferredToDate`.
    open var dateDecodingStrategy: JSONDecoder.DateDecodingStrategy

    /// The strategy to use in decoding binary data. Defaults to `.base64`.
    open var dataDecodingStrategy: JSONDecoder.DataDecodingStrategy

    /// The strategy to use in decoding non-conforming numbers. Defaults to `.throw`.
    open var nonConformingFloatDecodingStrategy: JSONDecoder.NonConformingFloatDecodingStrategy

JSONEncoder 在生成 JSON 的时候默认的格式是

{"userInfos":[{"age":18,"sex":true,"height":178.55999755859375,"userName":"小名"},{"age":18,"sex":false,"height":162.55999755859375,"userName":"小方"}]}

如果想要提高阅读体验, 可以设置属性 outputFormatting


open var outputFormatting: JSONEncoder.OutputFormatting

let jsonEncoder = JSONEncoder()
jsonEncoder.outputFormatting = .prettyPrinted

然后生成的JSON格式就会变成这样

{
  "userInfos" : [
    {
      "age" : 18,
      "sex" : true,
      "height" : 178.55999755859375,
      "userName" : "小名"
    },
    {
      "age" : 18,
      "sex" : false,
      "height" : 162.55999755859375,
      "userName" : "小方"
    }
  ]
}

JSON中 没有 Model 所需要的 Key

如果JSON的数据结构中,有些 Key - Value 不一定出现,Model中对应的要实用可选类型

JSON数据:

{
    "userInfos": [
        {
            "userName": "小名",
            "age": 18,
            "sex": true
        },
        {
            "userName": "小方",
            "age": 18,
            "height": 162.56,
            "sex": false
        }
    ]
}

Model对象:


struct UserList: Codable {

    let userInfos: [UserInfo]

    struct UserInfo: Codable {

        let userName: String
        let age: Int
        let height: Float?
        let sex: Bool
    }
}

参考: