Swift中没有了strong
, assign
, copy
关键字,对于所有的class
类型变量都默认采用了strong
类型,如果需要指定使用weak
,则需要添加weak
关键字修饰。
正是由于这种默认的strong
类型,在闭包中会引起循环引用,导致内存无法释放,为了能够在闭包(block
)中正常释放内存,需要对闭包中的self
使用弱引用,也就是不持有self
对象。
如下面的代码中:
class UserInfoViewController {
// ...
private lazy var updateInfoBlock: (Bool)->Void = {
return { succ in
if succ {
self.updateSuccess()
} else {
self.updateFailed()
}
}
}()
// ...
}
按照上面的代码写的话,就会引入循环引用。因为self
(也就是UserInfoViewController
的实例对象)持有updateInfoBlock
,而updateInfoBlock
又反过来持有对象self
,这将会导致两者的内存都无法释放。
这个时候,我们可以在updateInfoBlock
中使用关键字weak
对self
进行修饰,避免循环引用的产生。
代码如下:
class UserInfoViewController {
// ...
private lazy var updateInfoBlock: (Bool)->Void = {
return { [weak self] (succ) in
if succ {
self?.updateSuccess()
} else {
self?.updateFailed()
}
}
}()
// ...
}
这样,我们就可以避免循环引用了。但是我们注意到,在闭包内,self
变成了optional
类型,这是因为weak
修饰的对象在外界有可能释放,释放后就变成了nil
,所以这里要用optional
。
针对这个例子,我们知道,在updateInfoBlock
的有效期内,self
是不可能释放的,因为self
释放需要先释放updateInfoBlock
,在这种情况下,我们可以使用unowned
代替weak
关键字,这样的话,闭包内的self
对象仍旧是原来的类型,可以直接使用。
代码如下:
class UserInfoViewController {
// ...
private lazy var updateInfoBlock: (Bool)->Void = {
return { [unowned self] (succ) in
if succ {
self.updateSuccess()
} else {
self.updateFailed()
}
}
}()
// ...
}
只有当weak
修饰的对象(self
)本身对当前对象(updateInfoBlock
)有强引用,也就是持有当前对象时,才可以把weak
替换为unowned
,因为unowned
修饰的对象在释放后将会变成野指针,导致应用crash
。所以只有能够确保修饰的对象(self
)不会先于当前对象(updateInfoBlock
)释放时,才可以使用unowned
关键字。
网友评论