你有没有尝试过JSONSerialization.jsonObject(with:options:)
?
var jsonString = "{" +
"\"Language\": {" +
"\"Field\":[" +
"{" +
"\"Number\":\"976\"," +
"\"Name\":\"Test\"" +
"}," +
"{" +
"\"Number\":\"977\"," +
"\"Name\":\"Test\"" +
"}" +
"]" +
"}" +
"}"
var data = jsonString.data(using: .utf8)!
let json = try? JSONSerialization.jsonObject(with: data)
Swift 有时会产生一些非常奇怪的语法。
if let number = json?["Language"]??["Field"]??[0]?["Number"] as? String {
print(number)
}
JSON 对象层次结构中的所有内容最终都被包装为可选(即AnyObject?
). Array<T>
下标返回非可选T
。对于这个包含在可选数组下标中的 JSON,返回Optional<AnyObject>
。然而,Dictionary<K, V>
下标返回一个Optional<V>
。对于这个 JSON,下标返回看起来非常奇怪的Optional<Optional<AnyObject>>
(ie. AnyObject??
).
-
json
is an Optional<AnyObject>
.
-
json?["Language"]
返回一个Optional<Optional<AnyObject>>
.
-
json?["Language"]??["Field"]
返回一个Optional<Optional<AnyObject>>
.
-
json?["Language"]??["Field"]??[0]
返回一个Optional<AnyObject>
.
-
json?["Language"]??["Field"]??[0]?["Number"]
返回一个Optional<Optional<AnyObject>>
.
-
json?["Language"]??["Field"]??[0]?["Number"] as? String
返回一个Optional<String>
.
The Optional<String>
然后被使用if let
生成 a 的语法String
.
最后注意:迭代字段数组如下所示。
for field in json?["Language"]??["Field"] as? [AnyObject] ?? [] {
if let number = field["Number"] as? String {
print(number)
}
}
斯威夫特 4 更新
Swift 4 让这一切变得更容易处理。我们再次将从您的测试数据开始("""
让这变得更好)。
let data = """
{
"Language": {
"Field":[
{
"Number":"976",
"Name":"Test"
},
{
"Number":"977",
"Name":"Test"
}
]
}
}
""".data(using: .utf8)!
接下来,我们可以围绕 JSON 中使用的对象定义类。
struct Object: Decodable {
let language: Language
enum CodingKeys: String, CodingKey { case language="Language" }
}
struct Language: Decodable {
let fields: [Field]
enum CodingKeys: String, CodingKey { case fields="Field" }
}
struct Field: Decodable {
let number: String
let name: String
enum CodingKeys: String, CodingKey { case number="Number"; case name="Name" }
}
The CodingKeys
enum 是将结构体属性映射到 JSON 对象成员字符串的方式。这个映射是由自动完成的Decodable
.
现在解析 JSON 很简单。
let object = try! JSONDecoder().decode(Object.self, from: data)
print(object.language.fields[0].name)
for field in object.language.fields {
print(field.number)
}