Xcode8的调试技能又增长了一个黑科技:Memory Graph。简单的说就是能够在运行时将内存中的对象生成一张图。在现场的开发者听到了这个消息时响起了雷鸣般的掌声!咱们来看看前方记者发回的现场照片:swift
妈妈说不再用担忧引用循环啦!除非你是个瞎子。
api
typealias Action = () -> Void
class MyView: UIView {
var action: Action?
init(action: @escaping Action) {
self.action = action
super.init(frame: CGRect.zero)
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}复制代码
接着咱们在一个ViewController中初始化MyView,而且也保存为属性:闭包
class ViewController: UIViewController {
@IBOutlet weak var label: UILabel!
var myView: MyView?
override func viewDidLoad() {
super.viewDidLoad()
myView = MyView(action: testMethod)
}
func testMethod() {
label.text = "haha"
}
}复制代码
这vc的view上有一个label控件,在viewDidLoad时初始化myView,而且将自身的一个testMethod方法当作参数传给了myView。
testMethod中设置了自身label的text。
注意,划重点了!app
这个例子影射里开发中一个常见的场景:一个tableViewCell中有一个删除按钮,经过闭包将方法传进去,cell保存这个闭包;另外一方面这个闭包被调起后,删除某条数据后刷新数据源。ide
那么这么写会产生引用循环吗?函数
func testMethod() {
label.text = "haha"
}复制代码
核心在这段代码上,一个类的方法里设置自身的属性,会捕捉这个属性吗?这个地方能够写self,可是捕捉策略是unowned仍是strong呢?
这个闭包的实现是不能本身声明捕捉策略的:ui
因而就来验证一下。运行起来后,push这个ViewController后pop出去(记得要进行两次,好像只有一次Xcode有时不会启动分析)。
接着点击这个按钮:spa
有了这个图就很容易看出来了:myView保持了action,action保持了testMethod,testMethod中由于设置了vc的label因此也保持了VC。因此咱们能够肯定:方法中隐式的self的捕捉策略是strong。这样直接把方法传入子view中会引发引用循环。debug
这样就能够本身声明捕捉策略。这样的方式使用就和OC的block相似了:3d
myView = MyView(){ [unowned self] in
self.label.text = "haha"
}复制代码
不是直接传入testMethod方法,而是在传入的闭包中调用自身的方法:
myView = MyView(){ [unowned self] in
self.testMethod()
}复制代码
欢迎关注个人微博:@没故事的卓同窗