4-9 NSTimer的循环引用

NSTimer基本使用

[NSTimer scheduledTimerWithTimeInterval:0.01 target:self selector:@selector(timeredAction:) userInfo:nil repeats:YES];
- (void)timeredAction:(NSTimer*)timer {
    UIView *view = [[UIView alloc]initWithFrame:CGRectMake(100, 100, 100, 50)];
    view.backgroundColor = CNCbox_RandomColor;
    [self.view addSubview:view];
}

这时系统已经在运行定时器的方法了

NSTimer与RunLoop

我们创建好的NSTimer默认是运行在 NSDefaultRunLoopMode 下的 在ui滚动的时候系统是运行在 UITrackingRunLoopMode 下的。这时我们的Timer是不会执行方法的 只有在UI停止滚动的时候才会执行

想要在UI滚动的时候执行 需要加入到NSRunLoopCommonModes中
[[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];

通常情况下NSDefaultRunLoopMode和UITrackingRunLoopMode都已经被加入到了common modes集合中, 所以不论runloop运行在哪种mode下, NSTimer都会被及时触发

NSTimer 循环引用的问题

当我们的timer  repeats为NO的时候我们的timer运行完就回在即释放掉了

NSTimer的scheduledTimerWithTimeInterval: target: selector: userInfo: repeats:方法的最后一个参数为YES时,NSTimer会一直保留目标对象,直到自身失效才释放目标对象。执行完任务后,一次性的定时器会自动失效;重复性的定时器,需要主动调用invalidate方法才会失效。


- (void)viewWillDisappear:(BOOL)animated {
    [super viewWillDisappear:animated];
    [self.timer invalidate];
    self.timer = nil;
}

iOS10中,定时器的API新增了block方法,
    __weak typeof(self) weakself = self;
    _timer = [NSTimer timerWithTimeInterval:0.01 repeats:YES block:^(NSTimer * _Nonnull timer) {
        [weakself timeredAction:nil];
    }];

如何在子线程使用NSTimer 最好不要使用Run会导致控制器释放不掉

- (void)threadStart {
    __weak __typeof(self) weakSelf = self;
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        __strong __typeof(weakSelf) strongSelf = weakSelf;
        if (!strongSelf) return;
        strongSelf.thread1 = [NSThread currentThread];
        [strongSelf.thread1 setName:@"线程A"];
        strongSelf.threadTimer = [NSTimer scheduledTimerWithTimeInterval:1 target:strongSelf selector:@selector(timerAction) userInfo:nil repeats:YES];
        NSRunLoop *runloop = [NSRunLoop currentRunLoop];
        [runloop addTimer:strongSelf.threadTimer forMode:NSDefaultRunLoopMode];
        [runloop run];
    });
}

- (void)timerAction {
    
}

如果这个方法跟创建NSTimer不在同一个线程执行是无法将Timer 执行invalidate操作的
然后现在我们需要在thread1这个线程中执行这个操作,在这里写一个方法用于在子线程中调用此方法


- (void)cancelTimer{    
    if (self.threadTimer && self.thread1) {
        [self performSelector:@selector(cancel) onThread:self.thread1 withObject:nil waitUntilDone:YES];
    }
}

- (void)cancel {
    if (!self.threadTimer) return;
    [self.threadTimer invalidate];
    self.threadTimer = nil;
}

上面的一大堆会释放 self.threadTimer.但是因为手动开启子线程的运行循环 (这个是主线程的运行循环和子线程的运行循环唯一的不同点)
 run : 一旦调用这个方法开启子线程的运行循环,就不会停止
 一旦开启运行循环,相当于就开启了死循环



`可以释放的代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [super touchesBegan:touches withEvent:event];
    NSLog(@" touchesBegan ");
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(update) userInfo:nil repeats:YES];
        [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
        CFRunLoopRun();
    });
}



- (void)update {
    static int i = 0;
    NSLog(@"%d", i);
    i++;
    if (i > 4) {
        CFRunLoopStop(CFRunLoopGetCurrent());
        i = 0;
    }
    NSLog(@"update: %@",[NSThread currentThread]);
}

推荐使用GCD

    [self startGCDTimerCompletion:^{
        NSLog(@"startGCDTimerCompletion: %@",[NSThread currentThread]);
    }];
}


- (void)startGCDTimerCompletion:(void (^)(void))completion {
    if (_gcdTimer) return;
    NSTimeInterval period = 0.1; //设置时间间隔
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    _gcdTimer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
    dispatch_source_set_timer(_gcdTimer, dispatch_walltime(NULL, 0), period * NSEC_PER_SEC, 0);
    // 事件回调
    dispatch_source_set_event_handler(_gcdTimer, ^{
        NSLog(@"event_handler: %@",[NSThread currentThread]);
        dispatch_async(dispatch_get_main_queue(), ^{
            if (completion) completion();
        });
    });
    // 开启定时器
    dispatch_resume(_gcdTimer);
}

- (void)endGCDtimer {
    if (!_gcdTimer) return;
    dispatch_source_cancel(_gcdTimer);
    _gcdTimer = nil;
}

fire 和 fireDate

fireDate指的是 10秒以后开始执行

fire 指的是 马上执行

iOS | 小心NSTimer中的内存泄漏

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