关于NSTimer容易踩的坑

想了解NSTimer的坑我们应该先熟悉下NSTimer的原理也就是他是怎么实现定时器的

runloop处理三种事件,source 、timer 、oberseve (这个大家可以看下runloo的原理 http://blog.ibireme.com/2015/05/18/runloop/),所以timer的实现原理是跟runloop分不开的

问题1:NStimer的记时准确吗?为什么?

       我们都知道NSTimer必须加入在runoop的某个模式下他的定时才会起作用,Runloop在oc 层面暴露给我们的只有两种模式    和    模式,加入我们加入的是 模式那么在列表滑动的时候,runloop会切换到   那么在滑动过程中NSTmer到了改做操作的时候可能不执行这是 timer 为什么记时不准确的原因之一,这个原因当然大部分人也应该知道。其实还有另外一个原因.

      NStimer 并不是硬件记时,他是在加入runloo拍的时候后在runoop中注册了一系列的时间点,runloop会在每次循环的时候去检查是否有timer事件要处理,有就处理,当然这个runoop的循环也可能有timer 唤醒,这个时间点有个宽容度,如果在改时间点的宽容度范伟内就执行timer 事件,当然我所说的并不是因为这个宽容度说其记时不准确,timer依赖于runloop 那么当我们在改runloop所在的线程中做了一些耗时操作 比方说特别大的一个循环,那么这个耗时操作,那么等执行到timer 事件的时候 已经过了该注册的时间点,不在其宽容度范围内,那么这次的timer 事件将被丢弃,也会造成记时不准确。GCD 中dispatch_source_set_timer 也是个计时器,是依赖于手机硬件,不依赖于runoop的计时器(可以了解下)

问题2:NSTimer 持有的tagert的释放:

  NSTimer 是加在runoop中的所以runoop对timer会有个强持有关系,所以假如runoop一直存在 那么timer将不会自动释放必须手动释放。如下面

NSTimer *timer = [NSTimer timerWithTimeInterval:0.5 target:self selector:@selector(load) userInfo:nil repeats:YES];

[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

那么NStimer对tagert 也是强引用,所以在释放self的时候必须先调用

[timer invalidate] 然后self才能被释放。为了解决self的释放对 timer先停止的依赖。所以苹果在10.0的时候给NStimer 添加了带Block块的个方法,如下面这个:

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));

对过__weak的弱引用从而使得self 的释放不需要依赖timer 的 invalidate

那么我们在10.0 应该怎么兼容10.0之前的呢?可以通过Category的形式 然后 给timer 增加新方法。

具体的实现可以参照如下代码:

@interface NSTimer (QCBlock)

/**

*  剩余计数次数

*  Tip:此属性仅本类目中初始化方法创建的定时器有效!

*/

@property (assign, nonatomic) NSNumber *count;

/**

*  指定次数定时器

*

*  @param interval 回调时间间隔

*  @param count 重复次数  Tip1: 如果count == NSIntegerMax 则表示无限次

*                        Tip2: 如果count <= 0 则表示终止,否则表示具体的次数

*  @param callback 回调Block

*

*  @return NSTimer对象

*/

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval

count:(NSInteger)count

callback:(void (^)(NSInteger count))callback

finishback:(void (^)(void))finishback;

/**

*  启动定时器

*/

- (void)fireTimer;

/**

*  暂停定时器

*/

- (void)suspendTimer;

/**

*  终止定时器

*/

- (void)stopTimer;

@end

.m里面的实现

@implementation NSTimer (QCBlock)

- (NSNumber *)count {

return objc_getAssociatedObject(self, @selector(count));

}

- (void)setCount:(NSNumber *)count {

objc_setAssociatedObject(self, @selector(count), count, OBJC_ASSOCIATION_COPY_NONATOMIC);

}

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval

count:(NSInteger)count

callback:(void (^)(NSInteger))callback

finishback:(void (^)(void))finishback {

NSAssert(count > 0, @"The value of the parameter ‘count’ cannot be less than 0! If you want to represent unlimited, you can set the value of ‘count’ is ‘NSIntegerMax’.");

NSDictionary *dic = @{ @"callback" : [callback copy],

@"finishback" : [finishback copy] };

NSTimer *t = [NSTimer scheduledTimerWithTimeInterval:interval

target:self

selector:@selector(onTimerUpdateCountBlock:)

userInfo:dic

repeats:YES];

t.count = @(count);

return t;

}

+ (void)onTimerUpdateCountBlock:(NSTimer *)timer {

void (^callback)(NSInteger) = timer.userInfo[@"callback"];

timer.count = @([timer.count integerValue] - 1);

if (callback) {

callback([timer.count integerValue]);

}

if ([timer.count integerValue] == 0) {

void (^finishback)(void) = timer.userInfo[@"finishback"];

[timer stopTimer];

if (finishback) {

finishback();

}

}

}

- (void)fireTimer {

[self setFireDate:[NSDate distantPast]];

}

- (void)suspendTimer {

[self setFireDate:[NSDate distantFuture]];

}

- (void)stopTimer {

if (self.isValid) {

[self invalidate];

}

}

@end

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,924评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,781评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,813评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,264评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,273评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,383评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,800评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,482评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,673评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,497评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,545评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,240评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,802评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,866评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,101评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,673评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,245评论 2 341

推荐阅读更多精彩内容