将 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 对象的完整响应
首先使用
将数据转换为JSON对象
让json响应=尝试JSONSerialization.jsonObject(使用:数据,选项:.mutableContainers)作为!词典
获取用户列表字符串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输出到模型
结果:
我有这个 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 对象的完整响应
首先使用
将数据转换为JSON对象让json响应=尝试JSONSerialization.jsonObject(使用:数据,选项:.mutableContainers)作为!词典
获取用户列表字符串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输出到模型
结果: