Swift 3中可选项的奇怪行为



我在使用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
    }
}

最新更新