Swift: If let 语句无法处理空数组

Swift: If let statement failing to handle empty array

我有一个应用程序正在使用 Foursquare API 下载 JSON 数据。我正在使用 NSURLSession 和带有完成块方法的 dataTaskWithRequest 来获取数据。我得到的数据很好,但有时名为 groups 的嵌套数组可能为空。当我像下面这样解析 JSON 时,由于某种原因,我的条件语句没有像我期望的那样处理空数组。如果不是通过运行时错误说明:index 0 beyond bounds of empty array

,而不是将数组评估为空并继续执行 if let...else 语句的 "else" 部分
if let response: NSDictionary = data["response"] as? [String: AnyObject],
            groups: NSArray = response["groups"] as? NSArray,
                        // Error here \|/ (sometimes groups array is empty)
            dic: NSDictionary = groups[0] as? NSDictionary,
            items: NSArray = dic["items"] as! NSArray {

}

else {

    // Never gets here. Why?
    // IF the groups array is empty, then the if let should return 
    // false and drop down to the else block, right?
}

我是 Swift 的新手,谁能告诉我为什么会这样,我能做些什么来解决这个问题?谢谢

如果数组为空,您必须在 if let 语句外显式检查,因为

空数组永远不是可选的

if let response = data["response"] as? [String: AnyObject], groups = response["groups"] as? NSArray {
  if !groups.isEmpty {
    if let dic = groups[0] as? NSDictionary {
       items = dic["items"] as! NSArray
       // do something with items
       println(items)
    }
  }
} else ...

向下转换类型时可以省略所有类型注释

但是,您可以使用 where 子句执行检查,这适用于 Swift 1.2 和 2

if let response = data["response"] as? [String: AnyObject],
                  groups = response["groups"] as? [AnyObject] where !groups.isEmpty,
                  let dic = groups[0] as? NSDictionary,
                  items = dic["items"] as? NSArray {
   // do something with items
   println(items)
} else {...

在语句之间使用 && 运算符。

例如这不会崩溃:

var a = 4;
var c = 5;
var array = Array<Int>();

if  a > 2 &&  c > 10 && array[0] != 0 {

}

在尝试访问之前确保数组不为空:

if let response: NSDictionary = data["response"] as? [String: AnyObject],
    let groups: NSArray = response["groups"] as? NSArray where groups.count > 0 {
        if let dic: NSDictionary = groups[0] as? NSDictionary,
            let items: NSArray = dic["items"] as? NSArray {
             // Do something..
             return // When the 2nd if fails, we never hit the else clause
        }
}

// Do logic for else here
...