将 json 输出解码到模型

Decode json output to a model

我有这个 json 输出,我想使用 Codable 解析:

{
"success": true,
"total": 1,
"users": [
    {
        "user": {
            "id": "1",
            "fname": "admin",
            "lname": "admin",
            "login": "admin",
            "actif": "0",
            "last_connection_date": "2018-01-18 16:02:34"
        }
    }
],
"msg": ""
}

我只想从中提取用户的信息。 我的用户模型

import RealmSwift

class User: Object, Codable {

@objc dynamic var id: String = ""
@objc dynamic var fname: String = ""
@objc dynamic var lname: String = ""
@objc dynamic var login: String = ""

//    private enum CodingKeys : String, CodingKey {
//        case id = "users[0].user.id"
//        case fname = "users[0].user.fname"
//        case lname = "users[0].lname"
//        case login = "users[0].user.login"
//        case password = "users[0].user.password"
//    }

}

// Somewhere in my code
Alamofire.request(Path.userInformations(id: userId).rawValue).
responseJSON(completionHandler: { response in
       do {
            let user = try JSONDecoder().decode(User.self, from: response.data!)
            } catch (let error) {
                print(error.localizedDescription)
            }
       })

我尝试提取用户的对象,但未能成功将其转换为 Data 以将其提供给 JSONDecoder().decode() 方法。

回复 Vishal16 的评论

我试过你第一种方法。它似乎不起作用,因为我认为关键字 "user" 在用户对象之前。我试过添加一个新的结构来包装用户的对象,但没有解决它。

struct ResponseBody : Codable {
   var success : Bool?
   var total : Int?
   var users : [UserHolder]?
   var msg : String?
   var query_start : String?
   var query_end : String?
   var query_time : String?
   var paging : Bool?
}

struct UserHolder : Codable {
    var user: User?

    enum CodingKeys: String, CodingKey {

        case user = "user"

    }

    init(from decoder: Decoder) throws {
        let values = try decoder.container(keyedBy: CodingKeys.self)
        user = try values.decodeIfPresent(User.self, forKey: .user)
    }
}
class User: Object, Codable {

@objc dynamic var id: String = ""
@objc dynamic var fname: String = ""
@objc dynamic var lname: String = ""
@objc dynamic var login: String = ""

}

class Users: Object, Codable {

@objc dynamic var users: [User]

}

并用于解码

let user = try JSONDecoder().decode(Users.self, from: response.data!)

我认为应该可以解决问题。

另一种方法是,您必须将 response.data 转换为 Dictionary 才能挖掘到 user 对象。

我认为您的回复 class 结构应该是这样的:

import Foundation
struct ResponseBody : Codable {
var status : Bool?
var total : Int?
var users : [User]? //list of users 
var msg : String?

enum CodingKeys: String, CodingKey {

    case status = "status"
    case total = "total"
    case users = "users"
    case msg = "msg"
}

init(from decoder: Decoder) throws {
    let values = try decoder.container(keyedBy: CodingKeys.self)
    status = try values.decodeIfPresent(Bool.self, forKey: . status)
    total = try values.decodeIfPresent(Int.self, forKey: . total)
    users = try values.decodeIfPresent([User].self, forKey: . users)
    msg = try values.decodeIfPresent(String.self, forKey: . msg)
}
}

现在您将能够将 JSON 数据检索到对象

let jsonDecoder = JSONDecoder()
let response = try jsonDecoder.decode(ResponseBody.self, from: data)

for user in response.users {
   // user object is here
}

#edit

如果您不想解析对 JSON 对象的完整响应

  1. 首先使用

    将数据转换为JSON对象

    让json响应=尝试JSONSerialization.jsonObject(使用:数据,选项:.mutableContainers)作为!词典

  2. 获取用户列表字符串JSON然后将其转换为数据,然后将数据转换为用户列表对象

    if let responseBody = jsonResponse["users"] {

    let dataBody = (responseBody as!String).data(using: .utf8)! 如果让 obj = Utils.convertToArray(data: dataBody) { print(obj) // 用户对象列表 } }

听说是上面实现中使用的方法

class func convertToArray(data: Data) -> [AnyObject]? {
    do {
        return try JSONSerialization.jsonObject(with: data, options: []) as? [AnyObject]
    } catch {
        Constants.print(items: error.localizedDescription)
    }
    return nil
}

希望对您有所帮助。编码愉快 :)

听到的是适合您的工作代码 它在我的操场上运行良好。请看下面的截图

1.

2.

3.

解码json输出到模型

结果: