delegate 为什么应该是 weak 类型而不是strong类型

本文深入探讨了在Objective-C中使用Weak类型作为Delegate的原因,旨在避免循环引用导致的内存泄露问题,同时解释了Strong类型与Weak类型在Delegate应用上的区别及其对程序生命周期的影响。
delegate 为什么应该是 weak 类型而不是strong类型


循环引用
对象a创建并引用了对象b.对象b创建并引用了对象c.对象c创建并引用了对象b.
这时候bc的引用计数分别是21。当a不再使用b,调用release释放对b的所有权,因为c还引用了b,所以b的引用计数为1b不会被释放。b不释放,c的引用计数就是1c也不会被释放。从此,bc永远留在内存中。

打断这种循环引用:
如果c引用b的时候是用weak类型弱引用,那么a释放b时,b的引用计数器变为0,b被释放,c也被释放。


一个例子:

// A.m中某处 

B* b = [B alloc] init]; 

b.delegate = self

[self.view addSubview:b];

[b release];

原因1 

delegate没有必要是strong A负责创建B的,A的生命周期一定比B要长(B存在,A一定也存在;A存在,B不一定存在) 也就是说 B(实例b)存在的时候,A(的实例)一定存在, 所以没有必要用retain将引用计数+1assign足以 

原因2 

退一步假设存在这样的情况:AB先挂掉(A的实例先被release销毁) 那么当然希望是[a relase]后,A中的方法不再被B调用,但是使用retain时候,A还是没有被销毁,so A中的方法仍会被B调用,但是assgin就无此问题 

原因3 

避免循环引用 两个类互相强引用,referance counting永远不会是0,一直会占用内存,如果delegate是弱引用,那delegate释放掉之后,这个类也可以被释放


另一个例子:


一个UITableViewController 对象a通过retain获取了UITableView对象b的所有权,这个UITableView对象bdelegate又是a 如果这个delegateretain方式的,那基本上就没有机会释放这两个对象了。 循环引用而产生的内存泄露也是Instrument无法发现的

所以delegate必须设置为weak类型




参考:
iOS 5 ARC完全指南.pdf 
http://blog.youkuaiyun.com/diyagoanyhacker/article/details/6591593
http://www.cocoachina.com/ask/questions/show/93387


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值