带闭包的属性初始化

时间:2018-03-07 20:56:00

标签: swift closures strong-references

我正在研究ARC和强大的参考周期并遇到了我的这段代码:

class TestClass: UIView {
  let button: UIButton = {
    let view = UIButton()
    view.frame = CGRect(x: 50, y: 50, width: 200, height: 200)
    view.backgroundColor = .blue
    view.translatesAutoresizingMaskIntoConstraints = false
    view.setTitle("Button", for: .normal)
    view.addTarget(self, action: #selector(buttonClicked), for: .touchUpInside)
    return view
  }()

  @objc private func buttonClicked() {
    print("Clicked")
  }

  override init(frame: CGRect) {
    super.init(frame: frame)
    print("Object of TestClass initialized")
  }

  required init?(coder aDecoder: NSCoder) {
    fatalError("init(coder:) has not been implemented")
  }

  deinit {
    print("Object of TestClass deinitialized")
  }
}

在闭包内self方法中对addTarget的引用似乎没有创建强大的参考周期。

有人可以解释原因吗?

另外,我注意到如果从UIView删除继承,编译器会开始抱怨:Use of unresolved identifier 'self'

有人可以解释一下,为什么会在这种情况下发生,而不是在第一种情况下呢?

1 个答案:

答案 0 :(得分:1)

这是保留周期,因为self不是您认为的那样:)

具有初始值的属性即使在任何初始化程序运行之前也会“执行”,并且对于那些属性self指向此类型的更高阶函数:

(TestClass) -> () -> TestClass

因此,您并不真正访问该实例,而是访问类似static的方法,该方法会对具有默认值的所有属性进行初始化。这就是您没有保留周期的原因。

addTarget为其第一个参数接受Any?值,因此这违反了没有类型规则,因此编译器不会抱怨您没有在那里传递NSObject实例。 / p>

检查以后的行为 - 例如如果按钮被添加到UI层次结构并被轻击,会发生什么?有趣的是:运行时看到您将非对象作为目标传递并为目标和操作设置空值:

enter image description here