在做限時(shí)支付堂油,驗(yàn)證碼發(fā)送之類的功能時(shí)經(jīng)常需要使用NTimer來(lái)做定時(shí)器菠隆,但是NSTimer在invalidate之前會(huì)保留持有它target對(duì)象榄笙,導(dǎo)致targtet對(duì)象無(wú)法釋放舶得,即使在delloc中:
- (void)dealloc{
[self.timer invalidate];
self.timer = nil;
}
也無(wú)法釋放尝抖,因?yàn)閠arget對(duì)象一直被引用就不會(huì)進(jìn)入delloc方法。
- 所以最基本的方法就是在target對(duì)象需要釋放之前手動(dòng)去控制執(zhí)行計(jì)時(shí)器的invalidate方法窄潭,但有時(shí)計(jì)時(shí)器的invalidate方法并不是完全能控制調(diào)用的春宣,因?yàn)橐_定target對(duì)象在最后一個(gè)引用釋放之前調(diào)用計(jì)時(shí)器的invalidate方法,這通過(guò)代碼無(wú)法完全檢測(cè)出來(lái)嫉你。在查看了《Effective Objective-C 2.0:編寫(xiě)高質(zhì)量iOS與OS X代碼的52個(gè)有效方法》之后得知可以用塊來(lái)打破“保留環(huán)”月帝。
首先,在分類中添加下面這段代碼:
#import <Foundation/Foundation.h>
@interface NSTimer (Addtions)
/**
* 計(jì)時(shí)器動(dòng)作
*
* @param interval 時(shí)間
* @param block 事件
* @param repeats 是否重復(fù)
*/
+ (NSTimer *)sf_scheduledTimerWithTimeInterval:(NSTimeInterval)interval
block:(void(^)())block
repeats:(BOOL)repeats;
@end
#import "NSTimer+Addtions.h"
@implementation NSTimer(Addtions)
/**
* 計(jì)時(shí)器動(dòng)作
*
* @param interval 時(shí)間
* @param block 事件
* @param repeats 是否重復(fù)
*/
+ (NSTimer *)sf_scheduledTimerWithTimeInterval:(NSTimeInterval)interval
block:(void(^)())block
repeats:(BOOL)repeats{
return [self scheduledTimerWithTimeInterval:interval
target:self
selector:@selector(private_blockinvoke:)
userInfo:[block copy]
repeats:repeats];
}
+ (void)private_blockinvoke:(NSTimer *)timer{
void (^block)() = timer.userInfo;
if (block) {
block();
}
}
@end
在代碼中幽污,把要執(zhí)行的事件封裝成“塊”嚷辅,通過(guò)userInfo參數(shù)傳出去(uerInfo可用來(lái)存放“不透明值”),那么只要計(jì)時(shí)器有效距误,就會(huì)一直保留著它簸搞。這里有個(gè)地方要注意一下,傳遞block時(shí)需要copy一下准潭,把block拷貝到“堆”上趁俊,因?yàn)槎x塊的時(shí)候,所占的內(nèi)存區(qū)域是分配在棧中的惋鹅,這說(shuō)明则酝,塊只在定義它的那個(gè)范圍內(nèi)有效,通過(guò)拷貝就可以把塊從棧復(fù)制到堆上闰集,塊就可以在定義它的范圍之外使用。
現(xiàn)在計(jì)時(shí)器的target是NSTimer類對(duì)象般卑,而NSTimer類對(duì)象是個(gè)單例武鲁,是否持有它都無(wú)所謂了。
現(xiàn)在我們直接使用這個(gè)函數(shù)去創(chuàng)建計(jì)時(shí)器:
self.timer = [NSTimer sf_scheduledTimerWithTimeInterval:1.0
block:^{
[self timerAction];
} repeats:YES];
這時(shí)依然會(huì)形成“保留環(huán)”蝠检,因?yàn)閎lock保留了self對(duì)象沐鼠,self又持有timer屬性。所以在調(diào)用函數(shù)之前要使用weak引用來(lái)打破它:
__weak RegistVC *weakSelf = self;
self.timer = [NSTimer sf_scheduledTimerWithTimeInterval:1.0
block:^{
RegistVC *strongSelf = weakSelf;
[strongSelf timerAction];
} repeats:YES];
到這里叹谁,問(wèn)題就算解決了饲梭,當(dāng)self的最后一個(gè)引用將其釋放的時(shí)候,self就會(huì)被釋放了焰檩,如果忘記在delloc中調(diào)用計(jì)時(shí)器的invalidate方法憔涉,則weakSelf會(huì)變?yōu)閚il。在調(diào)試過(guò)程中也能看到對(duì)象delloc了析苫,而不是再持有一段時(shí)間甚至永遠(yuǎn)不會(huì)釋放兜叨。