我在使用Swift 3解析JSON数据时遇到了一个奇怪的行为。
do {
let json = try JSONSerialization.jsonObject(with: data!, options: []) as! NSDictionary
let items:[AnyObject] = (json["items"] as? [AnyObject])!
for item in items {
let id:String = item["id"] as! String
print("ID: (id)")
let info = item["volumeInfo"] as AnyObject
print(info)
let title = info["title"]
print(title)
}
} catch {
print("error thrown")
}
这会产生以下输出。请注意,info是可选的,但如果我试图打开它,它会说它不是可选的!脚本在let title = info["title"]
上崩溃,因此我无法访问标题键。此行为自Swift 2起改变。
ID: lbvUD6LUyV8C
Optional({
publishedDate = 2002;
publisher = "Sams Publishing";
title = "Java Deployment with JNLP and WebStart";
})
你可以这样做:
do {
let json = try JSONSerialization.jsonObject(with: data!) as! [String: Any]
let items = json["items"] as! [[String: Any]]
for item in items {
let id = item["id"] as! String
let info = item["volumeInfo"] as! [String: Any]
let title = info["title"] as! String
print(id)
print(info)
print(title)
}
} catch {
print("error thrown: (error)")
}
我可能会建议删除!
强制展开的代码(如果JSON不是您期望的形式,您真的希望它崩溃吗?),但希望这说明了基本思想。
info
的运行时类型是Optional<Something>
,但编译时类型(当您显式转换它时)是AnyObject
。编译器如何知道AnyObject
将恰好是Optional<Something>
,除非你告诉它(以强制转换的形式)?
试题:
let info = item["volumeInfo"] as AnyObject?
现在编译器知道它是一个Optional<AnyObject>
,所以你可以打开它
在Swift 3中,如果是数组或字典,编译器必须知道所有下标对象的类型。AnyObject
——在Swift 3中已被更改为Any
——是不够的。
既然你知道键volumeInfo
的值是一个字典强制转换,最好使用可选绑定
let info = item["volumeInfo"] as? [String:Any] {
print(info)
let title = info["title"] as! String
print(title)
}
应该这样做:
guard let json = try? JSONSerialization.jsonObject(with: data!, options: []) as! [String: AnyObject],
let items = json["items"] as! Array<AnyObject>? else {
return
}
for item in items {
let id = item["id"] as! String
if let info = item["volumeInfo"] as? [String: AnyObject] {
let title = info["title"] as! String
} else {
// do something
}
}