79

我是斯威夫特的新手。计算属性和设置为闭包的属性有什么区别?我知道计算属性每次都会重新计算。闭包有什么不同吗?IE

关闭:

var pushBehavior: UIPushBehavior = {
    let lazilyCreatedPush = UIPushBehavior()
    lazilyCreatedPush.setAngle(50, magnitude: 50)
    return lazilyCreatedPush
}()

计算:

var pushBehavior: UIPushBehavior {
    get{
        let lazilyCreatedPush = UIPushBehavior()
        lazilyCreatedPush.setAngle(50, magnitude: 50)
        return lazilyCreatedPush
    }
}
4

4 回答 4

130

简而言之,第一个是通过闭包初始化的存储属性,该闭包在初始化时仅被调用一次。第二个是计算属性,get每次引用该属性时都会调用其块。


存储属性的初始化闭包只调用一次,但您可以稍后更改存储属性的值(除非您用 替换varlet。当您想要封装代码以在单个简洁的代码块中初始化存储的属性时,这很有用。

但是,每次引用变量时都会调用计算属性的块。当您希望每次引用计算属性时都调用代码时,这很有用。通常,当您每次引用存储的属性时需要重新计算计算的属性(例如,从其他可能是私有的存储属性重新计算)时,您都会这样做。

在这种情况下,您无疑需要存储属性(第一个示例),而不是计算属性(第二个示例)。您可能不希望每次引用变量时都需要新的推送行为对象。


顺便说一句,在您的第一个示例中,您在内部引用它被延迟实例化。如果你想要这种行为,你必须使用lazy关键字:

lazy var pushBehavior: UIPushBehavior = {
    let behavior = UIPushBehavior()
    behavior.setAngle(50, magnitude: 50)
    return behavior
}()

但是,如果属性是static,它会自动延迟实例化。

于 2015-07-20T12:05:43.387 回答
7

关闭:

  //closure
    var pushBehavior: UIPushBehavior = {
        let lazilyCreatedPush = UIPushBehavior()
        lazilyCreatedPush.setAngle(50, magnitude: 50)
        return lazilyCreatedPush
    }()

第一次调用 pushBehavior 变量时,块执行并将值保存在 pushBehavior 变量中。之后,每当您调用 pushBehavior 时,都会返回这些值。

表示仅第一次执行并保存在此变量中的块代码。此外,您可以随时存储变量值,但之后会返回这些值,但如果您声明为“let”,则无法更改此值。

计算属性:

var pushBehavior: UIPushBehavior {
    get{
        let lazilyCreatedPush = UIPushBehavior()
        lazilyCreatedPush.setAngle(50, magnitude: 50)
        return lazilyCreatedPush
    }
}

在计算属性中,每当您调用 pushBehavior 变量时,都会执行此块并返回值。所以每次执行块。并且您不能将变量声明为 pushBehavior 变量的“let”关键字。

因此,您可以根据需要使用此代码。

于 2016-11-23T06:18:20.160 回答
5

主要区别在于您不能为计算属性分配某些内容,因为它没有设置器。在这种情况下,闭包只被调用一次,返回值被存储在变量中,所以如果结果不随时间改变,使用存储的变量比使用计算的变量更有效。

一般来说:只有在可以快速检索到值时才应使用计算属性。

旁注:如果您不更改/重新分配存储的变量,则应考虑将其设为常量(let

于 2015-07-20T12:02:32.177 回答
4

这不是一个答案,但值得一提的是:

  • 初始化完成必须知道存储属性的值。这通过默认或通过初始化发生。
  • 计算属性的值在被访问之前不会被计算
  • 延迟加载属性的值在被访问之前不会被定义

因此,对于计算变量和惰性变量,您都可以self毫无顾虑地访问或存储属性。

于 2018-06-13T20:34:06.890 回答