Swift getter覆盖非计算变量



在Objective-C中,能够进行真的很简单,也很好

 - (UIButton *)backButton{
      if(!_backButton){
           _backButton = [UIButton new];
      }
 }

然而,在Swift中,当重写属性getter时,它被称为计算变量,每次访问self.backButton时,都会重新计算该变量。下面的例子很好地说明了这一点:

private var backButton: UIBarButtonItem {
    let button = UIButton(frame: CGRect(x: 0, y: 0, width: self.view.frame.size.width*0.06, height: self.view.frame.size.width*0.06))
    button.setImage(UIImage(named: "back_arrow"), forState: UIControlState.Normal)
    button.rac_signalForControlEvents(UIControlEvents.TouchUpInside).subscribeNext {
        (next: AnyObject!) -> () in
        self.navigationController?.popViewControllerAnimated(true)
        return ()
    }
    println("Recalculating the button")
    let item = UIBarButtonItem(customView: button)
    return item
}

每次我访问self.backButton时都会调用println语句。此外,每次打印出来的内存地址也会发生变化。我知道这就是计算变量的本质,因为它们不存储在内存中。

这是一种类似的方式来复制Swift中Obj-C中完全相同的行为吗?我想要的只是一种实例化某个UI变量一次的方法,而不必将该代码放入初始化方法中。

最好的方法可能是将其创建为一个惰性变量,这样初始化器将在第一次访问时被调用一次。

lazy var backButton:UIBarButtonItem = {
    let button = ...
    return button
} ()

通过使用初始化器块,可以提供实例变量的复杂初始化。

您所指的是"惰性实例化"。您可以使用lazy关键字在swift中复制它:

private lazy var backButton: UIBarButtonItem = {
      let button = .....
      //....
      return button
 } ()

最新更新