何為循環(huán)引用倔矾?
簡單理解:對象A強(qiáng)引用B首启,對象B也強(qiáng)引用A峦萎;
何時會發(fā)生循環(huán)引用饮亏?
使用delegate、block的時候
Example:
新建一個Friend類如下:
#import <Foundation/Foundation.h>
typedef void(^FriendBLK)(NSString *name);
@interface Friend : NSObject
@property (nonatomic, strong) NSArray *arr;
@property (nonatomic, copy) FriendBLK block;
@end
#import"Friend.h"
@implementationFriend
- (id)init
{
if(self= [superinit]) {
self.arr=@[@111,@222,@333];
self.block= ^(NSString*name){
NSLog(@"name :%@ , block arr:%@", name, name);
};
}
returnself;
}
- (void)dealloc{
NSLog(@"Friend dealloc");
}
@end
然后使用的地方調(diào)用下:
Friend*f = [[Friendalloc]init];
f.block(@"yk-test");
當(dāng)f所在的runloop結(jié)束時忠藤,會發(fā)現(xiàn)控制臺打印
2016-09-07 10:20:15.264 RACDemo[2459:23583] name :yk-test , block arr:yk-test
2016-09-07 10:20:15.264 RACDemo[2459:23583] Friend dealloc
這表明f對象被釋放了挟伙,不存在內(nèi)存泄漏。然后我們稍微修改下代碼:
?再次運(yùn)行下模孩,發(fā)現(xiàn)只會打印
2016-09-07 10:27:24.216 RACDemo[2510:28277] name :yk-test , block arr:(
111,
222,
333
)
并不會調(diào)用dellaoc釋放self 也就是f對象未被釋放尖阔,即存在內(nèi)存泄漏;為了更好理解引用循環(huán)問題榨咐,我們添加一個延遲隊列介却,代碼如下:
然后發(fā)現(xiàn)f對象也是不釋放的,下面來分析原理:
外面f.block(@"yk-test");結(jié)束后块茁,需要釋放f對象齿坷,通知即將釋放,但是這時候block內(nèi)部還在強(qiáng)引用自己的arr?數(shù)組数焊。?Runloop發(fā)現(xiàn)f的retain count不為0永淌,于是乎f就不能釋放。等到block里面的引用self釋放之后昌跌,也就是retain count為0時仰禀,才能正在釋放;
解決方案:弱引用
__weak__typeof(self)wself =self; 或者使用@weakify() @strongify()
原理示意圖:
基本Block的問題解決了蚕愤。其實我們在使用delegate的時候也會出現(xiàn)同樣的問題答恶,所以使用delegate都用weak,而不能用strong萍诱;
以上就是日常開發(fā)的內(nèi)存泄漏之循環(huán)引用了悬嗓,有不對的地方請留言;