Swift中的weak和unowned关键字

Swift中默认采用强引用可能导致闭包中的循环引用,为了解决这个问题,可以使用`weak`和`unowned`关键字。`weak`用于创建不持有对象的弱引用,防止循环引用。当弱引用的对象被释放,它会自动变为`nil`。而`unowned`则假设对象在其引用范围内不会被释放,从而提供更高效的方式,但若对象提前释放,`unowned`引用将变为野指针,可能导致应用崩溃。在确保不会发生提前释放的情况下,可以安全地使用`unowned`替代`weak`。

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中使用关键字weakself进行修饰,避免循环引用的产生。

代码如下:

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关键字。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值