swift – 带闭包的属性初始化

我正在研究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")
  }
}

关闭内部的addTarget方法中引用self似乎没有创建强大的引用循环.

有人可以解释原因吗?

此外,我注意到如果我从UIView中删除继承,编译器开始抱怨:使用未解析的标识符’self’.

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

解决方法

这不是一个保留周期,因为自己不是你认为的那样:)

具有初始值的属性甚至在任何初始化程序运行之前都会“执行”,对于这些属性,self指向此类型的高阶函数

(TestClass) -> () -> TestClass

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

addTarget接受Any?它的第一个参数的值,所以这违反了没有类型规则,所以编译器不会抱怨你没有在那里传递NSObject实例.

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

enter image description here

相关文章

软件简介:蓝湖辅助工具,减少移动端开发中控件属性的复制和粘...
现实生活中,我们听到的声音都是时间连续的,我们称为这种信...
前言最近在B站上看到一个漂亮的仙女姐姐跳舞视频,循环看了亿...
【Android App】实战项目之仿抖音的短视频分享App(附源码和...
前言这一篇博客应该是我花时间最多的一次了,从2022年1月底至...
因为我既对接过session、cookie,也对接过JWT,今年因为工作...