iOS开发 • 实例——Hey, 定时器!

在现在很多app中,我们经常会看到轮播图,轮播广告等等,比如淘宝、京东商城app,他们都可以定时循环地播放广告、图片,背后的功臣之一就是今天的主角——定时器 NSTimer

简单地介绍了它的应用场景,接下来,说说此次要分享的技能点:

  • 定时器的常用方式
  • fire方法的正确理解
  • NSRunloopMode对定时器的影响
  • 子线程开启定时器
  • GCD定时器
  • 定时器引起的循环引用的解决思路

定时开始:

我创建一个HomeViewController,然后让他成为导航控制器的
RootViewController,在HomeViewControllerviewDidLoad调用定时器方法,如下代码:

#import "HomeViewController.h"

@interface HomeViewController ()

@end

@implementation HomeViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self regularTime];
}
/*
 定时器的常规用法
 */
- (void)regularTime
{
    //自动开启
    [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
    
}

- (void)timerAction
{
    NSLog(@"定时器:%s",__func__);
}

@end

运行结果:每隔一秒就打印一次。


运行结果.png

还有另外一种方式也可以开启定时器,那就是调用这个方法:
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

注意1:单独地写这一句代码,默认是不会开启定时器的,让我们看看苹果官方原文是怎么说的:“You must add the new timer to a run loop, using addTimer:forMode:. Then, after ti seconds have elapsed, the timer fires, sending the message aSelector to target. (If the timer is configured to repeat, there is no need to subsequently re-add the timer to the run loop.)
也就是说,需要添加到runloop中,手动开启。运行的结果同上。

注意2:在主线程中,runloop是默认开启,如果是在子线程中开启开启定时器,那么我们还需要手动开启runloop。运行的结果同上。

注意3: NSRunLoopCommonModesNSDefaultRunLoopMode优先级使用场景不同:一般都是默认模式。

  • 当使用NSTimer的scheduledTimerWithTimeInterval方法时,此时Timer会被加入到当前线程的RunLoop中,且模式是默认的NSDefaultRunLoopMode,如果当前线程就是主线程,也就是UI线程时,某些UI事件,比如UIScrollView的拖动操作,会将RunLoop切换成NSEventTrackingRunLoopMode模式,在这个过程中,默认的NSDefaultRunLoopMode模式中注册的事件是不会被执行的,也就是说,此时使用scheduledTimerWithTimeInterval添加到RunLoop中的Timer就不会执行。
  • 所以为了设置一个不被UI干扰的Timer,使用的模式是:NSRunLoopCommonModes,这个模式等效于NSDefaultRunLoopModeNSEventTrackingRunLoopMode的结合。(参考官方文档)
    代码如下所示:
- (void)viewDidLoad {
    [super viewDidLoad];
   //在主线程中开启定时器
    [self regularTime];
   //在子线程中开启定时器
//    [NSThread detachNewThreadWithBlock:^{
//     NSLog(@"%@",[NSThread currentThread]);
//     [self regularTime];
//    }];
}
- (void)regularTime
{
    timer = [NSTimer timerWithTimeInterval:1.0f target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
    //runloop中添加定时器
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
//在子线程中启用定时器必须开启runloop
//    [[NSRunLoop currentRunLoop] run];
}

- (void)timerAction
{
    NSLog(@"定时器:%s",__func__);
}

fire方法

简单说说fire方法,fire是火焰的意思,从字面意思可以联想到燃料、加速的意思,that’s right![timer fire]——>就是加速计时的意思,我们通过比如点击事件,来让定时器人为地加速计时,这个比较简单,这里就不多赘述。

GCD定时器

GCD定时器,通过创建队列、创建资源来实现定时的功能,如下代码所示:
注意:如果延迟2秒才开启定时器,那么dispatch_resume(gcdTimer)必须写在外面。

#import "HomeViewController.h"

@interface HomeViewController ()
{
    NSTimer * timer;
}

@end

@implementation HomeViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self gcdTimer:1 repeats:YES];
}

- (void)gcdTimer:(int)timerInterVal repeats:(BOOL)repeat
{
    //创建队列
    dispatch_queue_t queue = dispatch_queue_create("my queue", 0);
    //创建资源
    dispatch_source_t gcdTimer = dispatch_source_create(&_dispatch_source_type_timer, 0, 0, queue);
    dispatch_source_set_timer(gcdTimer,dispatch_time(DISPATCH_TIME_NOW, 0),1*NSEC_PER_SEC,0);
    
    dispatch_source_set_event_handler(gcdTimer, ^{
        if (repeat) {
            NSLog(@"重复了");
            [self timerAction];
        } else
        {
            //            [self timerAction];
            //            //调用这个方法,释放定时器
            //            dispatch_source_cancel(gcdTimer);
            //延迟两秒会出现什么情况呢?
            /*
             为何会调用两次?2秒之后再触发定时器后,耽搁了0.001秒去cancel,那定时器已经再次
             触发了,所以走了两次,解决的方法就是把cancel写在外面。
             */
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW,(int64_t)
            (timerInterVal*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                [self timerAction];
            });
            dispatch_source_cancel(gcdTimer);
        }
    });
    dispatch_resume(gcdTimer);
}

/*
 定时器的常规用法
 */
- (void)regularTime
{
    //自动开启
    [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector
    (timerAction) userInfo:nil repeats:YES];
    
}

- (void)timerAction
{
    NSLog(@"定时器:%s",__func__);
}
定时器循环引用的解决思路
  • 循环引用出现的场景:
    eg:有两个控制器A和B,A 跳转到B中,B开启定时器,但是当我返回A界面时,定时器依然还在走,控制器也并没有执行dealloc方法销毁掉。
  • 为何会出现循环引用的情况呢?
    原因是:定时器对控制器 (self) 进行了强引用。

先说简单的解决思路:
苹果官方为了给我们解决这个循环引用的问题,提供了一个定时器的新的自带方法:
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
代码如下:

- (void)regularTime
{
    //用苹果自带的方法,使用weakself就可以解决定时器循环引用问题
    __weak typeof(self)weakSelf = self;
    timer =  [NSTimer scheduledTimerWithTimeInterval:1.0f repeats:YES block:^(NSTimer * _Nonnull timer) {
        [weakSelf timerAction];
    }];
}

第二种思路:

  • 既然引发循环引用是因为Timer对self的强引用,那我让Timer不对self强引用,不就解决了。
  • 本人非常兴奋地实验了两种方法:timer=nil(失败)、__weak typeof(self)weakself = self(失败),都是调用系统自带的方法,如下代码:
- (void)regularTime
{
    //自动开启
    //timer置为nil或者__weak typeof(self)weakself = self也无法解决定时器循环引用问题
    __weak typeof(self)weakself = self;
    timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:weakself selector:
    @selector(timerAction) userInfo:nil repeats:YES];
}

既然如此,那该如何是好?
答案是:通过类扩展,自己改写NSTimer的类方法,在控制器中调用新的类方法,直接show the code:


NSTimer+HomeTimer.h中:

#import <Foundation/Foundation.h>

@interface NSTimer (HomeTimer)

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)timerInterval block:(void(^)())block repeats:(BOOL)repeat;
+ (void)timerAction:(NSTimer *)timer;
@end

NSTimer+HomeTimer.m中:

#import "NSTimer+HomeTimer.h"

@implementation NSTimer (HomeTimer)

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)timerInterval block:(void(^)())block repeats:(BOOL)repeat
{
    return [self timerWithTimeInterval:timerInterval target:self selector:@selector(timerAction:) userInfo:block repeats:YES];
}

+ (void)timerAction:(NSTimer *)timer
{
    void (^block)() = [timer userInfo];
    if (block) {
        block();
    }
}
@end

类扩展写好之后,在控制器中调用,重写类方法,让定时器对NSTimer类强引用,类是没有内存空间的,就没有循环引用,跟苹果提供的新方法是类似的处理方式,如下代码和运行结果所示:

#import "HomeTimerViewController.h"
#import "NSTimer+HomeTimer.h"

@interface HomeTimerViewController ()
{
    NSTimer * timer;
}
@end

@implementation HomeTimerViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self regularTime];
    self.view.backgroundColor = [UIColor greenColor];
}

- (void)regularTime
{
    __weak typeof(self)weakSelf = self;
    timer = [NSTimer timerWithTimeInterval:1.0f block:^{
        [weakSelf timerAction];
    } repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
}

- (void)timerAction
{
    NSLog(@"定时器:%s",__func__);
}

- (void)dealloc
{
    NSLog(@"%s",__func__);
}

@end
运行结果.png

定时结束:用时2小时32分钟

总结:

我之前在开发app的时候,对定时器更多是会用的层次,经过这次的深入学习,对定时器的原理有了更深入的理解、认识,技术的提升,很多时候都是基础知识的延伸,对原理理解透彻,很多东西就可以举一反三,全部都通了,希望对自己和各位同道人有所帮助。

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

推荐阅读更多精彩内容