循环引用问题:
将一个NSTimer对象作为一个控制器的的属性,这时当前VC对NSTimer对象进行了一次强引用。在创建NSTimer对象的时候,NSTimer对象又将当前VC作为自己的target,这时NSTimer对象对当前VC进行了一次强引用,这样就造成了NSTimer和当前VC的循环引用,从而让VC和NSTimer都无法释放,最终导致内存泄漏。
通常代码:
我们可以为NSTimer创建一个分类,在分类中添加一个创建NSTimer对象的方法,以避免循环引用的问题。
NSTimer分类代码:
.m中的代码
可复制代码:
+ (NSTimer *)resolve_scheduledTimerWithTimeInterval:(NSTimeInterval)inerval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block{
return [NSTimer scheduledTimerWithTimeInterval:inerval target:self selector:@selector(resolve_blcokInvoke:) userInfo:[block copy] repeats:repeats];
}
+ (void)resolve_blcokInvoke:(NSTimer *)timer {
void (^block)(NSTimer *timer) = timer.userInfo;
if (block) {
block(timer);
}
}
使用代码:
可复制代码:
weakifySelf
self.timer = [NSTimer resolve_scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer *timer) {
strongifySelf
[self doThings];
NSLog(@"----------------");
}];
关于weakSelf和strongSelf:
#define weakifySelf \
__weak __typeof(&*self)weakSelf = self;
#define strongifySelf \
__strong __typeof(&*weakSelf)self = weakSelf;
demo地址:https://gitee.com/liangsenliangsen/nstimer_loop_reference.git
2019.2.1新增:
iOS10之后新增的方法里有两个创建NSTimer对象的方法可以解决NSTimer不释放的问题。
看看几个方法的区别:
注意:利用scheduledTimerWithTimeInterval:方法创建的NSTimer对象不需要手动添加到NSRunLoop中就能使用,而利用timerWithTimeInterval:方法创建的NSTimer对象需要手动添加到NSRunLoop中才能使用。
本篇文章到这里就结束了,愿大家加班不多工资多,男同胞都有女朋友,女同胞都有男朋友。😊