可选的绑定、捕获引用和闭包 [可能的错误]

Optional Binding, Capturing References and Closures [possible bug]

我一直在努力让自己熟悉 Swift,但我最近遇到了这个涉及可选绑定和在闭包上下文中捕获引用的特殊问题。

给定以下声明(代码为了清楚起见进行了缩写;完整代码在末尾提供):

class Thing
{
      func giveNameIfSize() -> String? {
            /.../
      }
}

typealias IterationBlock = (Thing) -> Bool

class Iterable
{
     func iterate(block: IterationBlock) {
            /.../
     }
}

Iterable 对象可以存储 Things 的集合,并可以使用闭包遍历它们。现在假设我们想要找到具有 size 属性 设置的 Thing 对象的名称。我们可以这样做:

var name: String?
iterable.iterate { (someThing) -> Bool in
    /* 
      Assigning the result of the giveNameIfSize() function to 
      a variable 'name' declared outside of the closure and
      captured by reference 
    */
    if name = someThing.giveNameIfSize() { 
        return true
    }
    return false
}

但是,上面的代码会生成以下编译器错误:

Cannot assign to immutable value of type 'String?'

奇怪的是,当我们在可选绑定中使用另一个变量时问题就消失了:

iterable.iterate { (someThing) -> Bool in
    if var tempResult = someThing.giveNameIfSize() {
        name = tempResult
        return true
    }
    return false
} /* COMPILES AND RUNS */

如果我们在可选绑定之外给外部声明的变量name赋值,问题也解决了:

iterable.iterate { (someThing) -> Bool in
    name = someThing.giveNameIfSize()
    if name != nil {
        return true
    }
    return false
} /* ALSO COMPILES AND RUNS */

完整源代码here.

观察:使用 Swift 1.2 而非 2.0 测试此代码。

====

这是一个错误吗?或者我错过了什么?

您收到的错误有点误导,但这里的根本问题是您正在尝试执行 Swift 不支持的操作。在Swift中,不能在条件中使用赋值结果。

也就是说,您提出的两种替代方法都可以使用,尽管我倾向于认为这两种方法中的第一种要多一些 Swifty。