你试过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>>(即AnyObject??)。
-
json 是 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>。
然后if let 语法使用Optional<String> 生成String。
最后说明:迭代字段数组如下所示。
for field in json?["Language"]??["Field"] as? [AnyObject] ?? [] {
if let number = field["Number"] as? String {
print(number)
}
}
Swift 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" }
}
CodingKeys 枚举是结构属性映射到 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)
}