引子
我們今天不講Block這個語法,因為作為一個合格的開發(fā)者,如果你不知道,那么實在是說不過去。接下來我就當大家都知道了,如果確實不懂,建議先去看資料吧。
Block最主要的一個注意事項,就是retain cycle(循環(huán)引用問題),這會直接導(dǎo)致內(nèi)存問題。一般的處理方式就是在block內(nèi)使用weak 作為self指針的引用。這樣block內(nèi)就不會出現(xiàn)對self的強引用。
1. weakSelf和strongSelf的使用
請看下面這段代碼
__weak __typeof(self)weakSelf = self;
AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {
__strong __typeof(weakSelf)strongSelf = weakSelf;
strongSelf.networkReachabilityStatus = status;
if (strongSelf.networkReachabilityStatusBlock) {
strongSelf.networkReachabilityStatusBlock(status);
}
};
這段代碼來自于大家非常熟悉的AFNetwork框架中,功能是監(jiān)測網(wǎng)絡(luò)的,當然這個不是主要點。大家來看語法。
- 首先在block的最外層,定義一個weakSelf
- 在block在定義一個strongSelf,形成對weakSelf的引用,類型是__strong
- 在block內(nèi)多次使用strongSelf 調(diào)用函數(shù)
原因
估計有的朋友都沒想到還需要用strongSelf。這么寫的原因就在于,如果在block內(nèi)多次使用weakSelf進行調(diào)用函數(shù)或者實例變量,有可能會造成某一次調(diào)用的過程中weakSelf已經(jīng)失效。例子
__weak __typeof__(self) weakSelf = self;
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
[weakSelf do1];
[weakSelf do2];
});
do2在執(zhí)行的過程中可能已經(jīng)被釋放,失效了。
2. block都會產(chǎn)生retain circle嗎
答案是No,首先大家要理解retain circle產(chǎn)生的原因:A持有B,B持有A,依據(jù)OC的引用計數(shù)retain、release規(guī)則,雙方各自被對方持有,誰也無法正常釋放,循環(huán)引用就產(chǎn)生了。
A
...
-(void)dealloc{
[B release];
[super dealloc];
}
B
...
-(void)dealloc{
[A release];
[super dealloc];
}
上面的代碼一目了然,釋放A必須先釋放B,但是B的dealloc中又需要釋放A,誰也跑不掉了。
回到正題
[UIView animateWithDuration:duration animations:^{
[self.superview layoutIfNeeded];
}];
這種只是block內(nèi)引用了self,單向引用,不會產(chǎn)生retain circle,大膽的使用self。
</br>
dispatch_async(dispatch_get_main_queue(), ^{
self.headerImageView = image;
});
dispatch_async不知道self什么時候釋放,因此在調(diào)用的時候,先retain self一次,等到代碼執(zhí)行結(jié)束,在調(diào)用release,因此實際上并沒有增加self的引用計數(shù),也不需要擔心retain circle。