本人有若干成套学习视频, 可试看! 可试看! 可试看, 重要的事情说三遍 包含Java
, 数据结构与算法
, iOS
, 安卓
, python
, flutter
等等, 如有需要, 联系微信tsaievan
.
Swift在闭包的使用过程中有可能会引起循环引用, 这和OC中的循环引用是类似的.
现在假设我有两个控制器, 点击第一个控制器的跳转会跳转到第二个控制器, 第二个控制器pop回来的时候会deinit
控制器的简单跳转
我现在仅仅在B控制器里面写如下代码:
- 先定义一个闭包属性
//---------1. 先定义一个闭包属性 ---------
/* 这个闭包有一个String类型的参数, 无返回值 */
var finishedCallBack:((_ str: String) -> ())?
- 定义一个函数, 函数的参数为跟闭包属性类型相同的参数, 然后再把闭包参数赋值给属性
func demo(finished: @escaping (_ str: String) -> ()) {
self.finishedCallBack = finished
}
- 定义一个函数, 函数内部执行这个闭包
func secondDemo() {
if let finishedHandler = self.finishedCallBack {
finishedHandler("secondDemo is running")
}
}
- 最后, 在viewDidLoad中分别执行
demo()
和secondDemo()
两个函数- 执行
demo()
的时候是给闭包赋值 - 执行
secondDemo()
的时候是调用闭包
- 执行
override func viewDidLoad() {
super.viewDidLoad()
demo { (string) in
print("\(string)----\(self.view!)")
}
secondDemo()
}
如以上代码所示:
-
我在定义这个闭包的时候, 闭包内部引用了
self
, 那么闭包就对self进行了一个copy操作, 即闭包持有控制器 -
这个闭包又赋值给了闭包属性, 所以闭包属性对控制器有一个强引用
-
闭包属性又被控制器强引用, 所以他们之间构成了一个循环引用
如下图所示:
闭包的循环引用这个时候, 我在第二个控制器的页面pop回去的时候, deinit方法不会调用:
看控制台的打印:
循环引用后控制台的打印结果如何解除循环引用呢?
下面提供三个方法:
- 方法一: OC的方法, 在OC中,可以是有
__weak typeof(self)weakSelf = self;
在Swift中, 也有类似的方法:
override func viewDidLoad() {
super.viewDidLoad()
/* 方法一, OC的方法 */
weak var weakSelf = self
demo { (string) in
if let wSelf = weakSelf {
print("\(string)----\(wSelf.view)")
}
}
secondDemo()
}
Swift中就是weak var weakSelf = self
这句代码
- 方法二: [weak self]修饰闭包
override func viewDidLoad() {
super.viewDidLoad()
/* 方法二: [weak self]修饰闭包 */
demo {[weak self] (string) in
if let weakSelf = self {
print("\(string)----\(weakSelf.view)")
}
}
secondDemo()
}
- 方法三: [unowned self]修饰闭包
super.viewDidLoad()
/* 方法三: [unowned self]修饰闭包 */
demo {[unowned self] (string) in
print("\(string)----\(self.view)")
}
secondDemo()
}
注意: 方法二和方法三的区别在于, 当方法二中的控制器被销毁时, self指针会指向nil, 而当方法三中的控制器被销毁时, self指针是不会指向nil的,这时候就会形成野指针, 因此, 第三种方法解除循环引用是不推荐的, 有可能会引起一些问题
解除循环引用之后, 控制台的打印显示: deinit方法被调用了
解除循环引用后的控制台打印
网友评论