RAC框架源码解析之NSTimer

1、RunLoop基础知识

首先我们先简单的来说一下,我们都知道程序启动的时候去走进main函数,在mainAppDelegate作为入口并遵守代理,这样就会进入AppDelegate的代理方法中,如didFinishLaunchingWithOptions等方法。当走进main函数的时候,系统默认会开启一个运行循环并且是死循环也就是我们这节要说的RunLoop

那RunLoop它干了什么事?
1、负责监听事件(网络、时钟、触摸等)
2、如果没有事件发生,它会进入休眠状态
3、渲染界面,在一次循环中进行所有的渲染

1、NSTimer

NSTimer是苹果官方封装的一个倒计时类,使用起来比较简单。

//
//  ViewController.m
//  RAC_Timer
//
//  Created by JM on 2018/4/21.
//  Copyright © 2018年 JM. All rights reserved.
//

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerDown) userInfo:nil repeats:YES];
}

- (void)timerDown {
    static int i = 0;
    i++;
    NSLog(@"%d", i);
}

@end

控制台打印结果

2018-04-21 23:42:12.800474+0800 RAC_Timer[49953:2092718] 1
2018-04-21 23:42:13.801108+0800 RAC_Timer[49953:2092718] 2
2018-04-21 23:42:14.800973+0800 RAC_Timer[49953:2092718] 3
2018-04-21 23:42:15.800927+0800 RAC_Timer[49953:2092718] 4
2018-04-21 23:42:16.800965+0800 RAC_Timer[49953:2092718] 5
2018-04-21 23:42:17.800378+0800 RAC_Timer[49953:2092718] 6
2018-04-21 23:42:18.800847+0800 RAC_Timer[49953:2092718] 7
2018-04-21 23:42:19.799809+0800 RAC_Timer[49953:2092718] 8
2018-04-21 23:42:20.800715+0800 RAC_Timer[49953:2092718] 9
2018-04-21 23:42:21.799573+0800 RAC_Timer[49953:2092718] 10

2、NSTimer(RunLoop)

下面我们用RunLoop的方式来添加一下

下面有一个侧重点
1、NSDefaultRunLoopMode 为默认模式
2、UITrackingRunLoopMode 为UI模式,也就是说如果你创建该类型的RunLoop,它就只能被UI事件唤醒
3、NSRunLoopCommonModes 为占位模式,也就是说只要设置了该模式它就会同时通知默认模式以及UI模式

//
//  ViewController.m
//  RAC_Timer
//
//  Created by JM on 2018/4/21.
//  Copyright © 2018年 JM. All rights reserved.
//

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
//    [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerDown) userInfo:nil repeats:YES];

    NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerDown) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
}

- (void)timerDown {
    static int i = 0;
    i++;
    NSLog(@"%d", i);
}

@end

控制台打印结果

2018-04-22 01:19:16.039742+0800 RAC_Timer[51955:2174228] 1
2018-04-22 01:19:17.039835+0800 RAC_Timer[51955:2174228] 2
2018-04-22 01:19:18.039767+0800 RAC_Timer[51955:2174228] 3
2018-04-22 01:19:19.039715+0800 RAC_Timer[51955:2174228] 4
2018-04-22 01:19:20.039490+0800 RAC_Timer[51955:2174228] 5
2018-04-22 01:19:21.038807+0800 RAC_Timer[51955:2174228] 6
2018-04-22 01:19:22.039854+0800 RAC_Timer[51955:2174228] 7
2018-04-22 01:19:23.038787+0800 RAC_Timer[51955:2174228] 8
2018-04-22 01:19:24.039777+0800 RAC_Timer[51955:2174228] 9
2018-04-22 01:19:25.039236+0800 RAC_Timer[51955:2174228] 10

在开发中我们常常会把耗时操作放到子线程,像定时器这种东西我们也一般都是放在子线程的,但是会发现NSTimer在子线程中无效,我们对创建定时器的代码做如下修改:

    //创建子线程
    NSThread *thread = [[NSThread alloc] initWithBlock:^{
        NSLog(@"%@",[NSThread currentThread]);
        NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerDown) userInfo:nil repeats:YES];
        [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
    }];
    
    //启动线程
    [thread start];

控制台打印

2018-04-22 01:32:10.741969+0800 RAC_Timer[52318:2191101] <NSThread: 0x60400046e740>{number = 3, name = (null)}

在上述代码中我们发现并没有执行定时器方法,这是为什么呢!因为在子线程代码块结束的时候该线程被销毁了,所以就不会进入定时器方法也就是timerDown中。

那么该如何解决?
在上面的代码中我们发现有把定时器放在RunLoop中,但为什么还是失效呢!因为手动创建RunLoop默认是不运行的,我们需要运行RunLoop,如下

    //创建子线程
    NSThread *thread = [[NSThread alloc] initWithBlock:^{
        NSLog(@"%@",[NSThread currentThread]);
        NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerDown) userInfo:nil repeats:YES];
        [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
        //开启RunLoop循环
        [[NSRunLoop currentRunLoop] run];
    }];
    
    //启动线程
    [thread start];

控制台打印

2018-04-22 01:41:14.998636+0800 RAC_Timer[52531:2203640] <NSThread: 0x604000272040>{number = 3, name = (null)}
2018-04-22 01:41:16.002548+0800 RAC_Timer[52531:2203640] 1
2018-04-22 01:41:17.000635+0800 RAC_Timer[52531:2203640] 2
2018-04-22 01:41:18.000488+0800 RAC_Timer[52531:2203640] 3
2018-04-22 01:41:19.001231+0800 RAC_Timer[52531:2203640] 4
2018-04-22 01:41:20.003273+0800 RAC_Timer[52531:2203640] 5
2018-04-22 01:41:21.001323+0800 RAC_Timer[52531:2203640] 6
2018-04-22 01:41:22.001346+0800 RAC_Timer[52531:2203640] 7
2018-04-22 01:41:23.001389+0800 RAC_Timer[52531:2203640] 8
2018-04-22 01:41:24.004123+0800 RAC_Timer[52531:2203640] 9
2018-04-22 01:41:25.001335+0800 RAC_Timer[52531:2203640] 10

3、GCD实现倒计时

//
//  ViewController.m
//  RAC_Timer
//
//  Created by JM on 2018/4/21.
//  Copyright © 2018年 JM. All rights reserved.
//

#import "ViewController.h"

@interface ViewController ()

/** timer */
@property (nonatomic, strong) dispatch_source_t timer;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    //GCD设置timer
    dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_global_queue(0, 0));
    
    //GCD时间
    dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, 1.0 * NSEC_PER_SEC, 0);
    
    dispatch_source_set_event_handler(timer, ^{
        NSLog(@"%@", [NSThread currentThread]);
    });
    
    //启动
    dispatch_resume(timer);
    
    _timer = timer;

}
@end

控制台打印

2018-04-22 23:33:26.233998+0800 RAC_Timer[57174:2326547] <NSThread: 0x600000467180>{number = 3, name = (null)}
2018-04-22 23:33:27.234961+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:28.234041+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:29.234098+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:30.234248+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}

4、RAC实现倒计时

//rac实现倒计时
    [[RACSignal interval:1.0 onScheduler:[RACScheduler scheduler]] subscribeNext:^(NSDate * _Nullable x) {
        NSLog(@"%@", [NSThread currentThread]);
    }]

控制台打印

2018-04-22 23:33:26.233998+0800 RAC_Timer[57174:2326547] <NSThread: 0x600000467180>{number = 3, name = (null)}
2018-04-22 23:33:27.234961+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:28.234041+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:29.234098+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}
2018-04-22 23:33:30.234248+0800 RAC_Timer[57174:2326550] <NSThread: 0x604000467680>{number = 4, name = (null)}

demo源代码已放置GitHub地址https://github.com/JunAILiang/RAC_Demo

联系我:
qq: 1245424073
微信: liujunmin6980

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

推荐阅读更多精彩内容

  • 1 Runloop机制原理 深入理解RunLoop http://www.cocoachina.com/ios/2...
    Kevin_Junbaozi阅读 4,006评论 4 30
  • iOS刨根问底-深入理解RunLoop 概述 RunLoop作为iOS中一个基础组件和线程有着千丝万缕的关系,同时...
    reallychao阅读 821评论 0 6
  • Runloop 是和线程紧密相关的一个基础组件,是很多线程有关功能的幕后功臣。尽管在平常使用中几乎不太会直接用到,...
    jackyshan阅读 9,857评论 10 75
  • 概述 RunLoop作为iOS中一个基础组件和线程有着千丝万缕的关系,同时也是很多常见技术的幕后功臣。尽管在平时多...
    sumrain_cloud阅读 946评论 0 5
  • 小的时候每次我看到电视里演记录贫困地区人们生活的场景,心里总不是滋味。 那时候我就想,有没有一种方法,能够让穷人变...
    谢唠钱呗阅读 217评论 0 0