RunLoop与Timer以及常用Mode

创建一个新的项目就不说了,在主控制器上添加一个可以滑动的UI控件,这边我选择UITextView

直接上代码,代码已经注释的很清楚了

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    _finished = NO;
    /*创建一个定时器(以这种方式创建的定时器会被自动添加到当前的RunLoop中并且自动开始计时,很显然当前的RunLoop的Mode是NSDefaultRunLoopMode)*/
    //scheduledTimerWithTimeInterval开头设置的定时器都会自动添加到当前RunLoop的NSDefaultRunLoopMode下
    NSTimer *timer1 = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timeMethod1) userInfo:nil repeats:YES];
    //在界面上添加一个UITextView,里面设置足够多的文本以让它可以滑动,当我们滑动文本框的时候当前的的RunLoop的Mode会切换到UITrackingRunLoopMode用于处理UI的滚动等时间,这时timer因为是设置在NSDefaultRunLoopMode下才计时,故此时timer会暂停计时,知道处理完UI滚动等事件RunLoop再次切换回NSDefaultRunLoopMode这时timer才会继续计时.
    //NSRunLoopCommonModes表示的是NSDefaultRunLoopMode或UITrackingRunLoopMode,也就是说如果把timer设置在NSRunLoopCommonModes下,当RunLoop的Mode为NSDefaultRunLoopMode或UITrackingRunLoopMode计时器都可以正常计时
    
    //创建一个定时器,与上面定时器不同的是该定时器并不会自动添加到当前的RunLoop下,并且不会自动开始计时(可以运行一下,就知道timeMethod2)
    NSTimer *timer2 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod2) userInfo:nil repeats:YES];
    //如果我们手动开始计时,你回发现@"timeMethod2"只被打印一次就停止了,那时因为timer2并没有被添加到任何的RunLoop中
    [timer2 fire];
    
    //在创建一个定时器,与timer2不同的是我们把timer3手动添加到当前NSRunLoop中,此时计时器就会自动开始计时无需手动调用[timer3 fire]
    NSTimer *timer3 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod3) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer3 forMode:NSDefaultRunLoopMode];
    [timer3 fire];//此时计时器就会自动开始计时无需手动调用
    
    //上面timer1与timer3都是添加到当前的RunLoop中去,并且是在NSDefaultRunLoopMode下,此时只要滑动界面的UI,这两个定时器timer1与timer3都会暂停计时,为了让计时器在UI滚动时也可以正常计时,我们试着把计时器添加在当前RunLoopMode的UITrackingRunLoopMode模式下
    NSTimer *timer4 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod4) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer4 forMode:UITrackingRunLoopMode];
    //此时UI滚动时也timer4正常计时,现在在timeMethod4添加[NSThread sleepForTimeInterval:1.0];来模拟处理耗时操作,运行,在滑动UI你会发现虽然计时可以正常计时,但是明显感到了UI滚动时的卡顿,那时因为你在UITrackingRunLoopMode模式下处理了耗时的操作,此时的RunLoop纪要处理UI滚动也处理耗时操作自然会卡顿,显然不妥
    
    //解决timer4的问题,创建一个定时器,并且把他放到子线程中去,子线程我们这边使用继承自NSThread的自定义线程PJThread
    PJThread *pjThread1 = [[PJThread alloc] initWithBlock:^{
        NSTimer *timer5 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod5) userInfo:nil repeats:YES];
    }];
    //线程跑起来
    [pjThread1 start];
    //pjThread1执行后我们并没有看到"timeMethod5"被打印,并且打印出"PJThread 被销毁了!",说明pjThread1执行完就被销毁了,而timer5没有计时打印出"timeMethod5"是因为他没有被添加到任何RunLoop下
    
    //解决pjThread1与timer5的问题,我们在创建pjThread2与timer6
    PJThread *pjThread2 = [[PJThread alloc] initWithBlock:^{
        NSTimer *timer6 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod6) userInfo:nil repeats:YES];
        //[NSRunLoop currentRunLoop]会获取当前的RunLoop,**如果当前没有则会创建一个**
        NSRunLoop *runLoop1 = [NSRunLoop currentRunLoop];
        //在NSRunLoopCommonModes下也是可以的,但是苹果官方推荐NSDefaultRunLoopMode,在runLoop1下已经不会受到UI滚动的影响了
        [runLoop1 addTimer:timer6 forMode:NSDefaultRunLoopMode];
        //run之后代码会一直死循环在此处不会往下执行,所以"runLoop1执行之后!"是不会被打印的
        [runLoop1 run];
        NSLog(@"runLoop1执行之后!");
    }];
    [pjThread2 setName:@"pjThread2"];
    //线程跑起来
    [pjThread2 start];
    //此时timer6可以正常计时了,并且pjThread2也没有被销毁了,滑动UI也不会卡顿了,timer6在UI滚动时也可以正常计时,处理耗时操作,这是因为[runLoop1 run]之后代码会一直死循环在此处不会往下执行,所以"runLoop1执行之后!"是不会被打印的,并且pjThread2也不会被销毁
    
    __weak typeof(self) weakSelf = self;
    PJThread *pjThread3 = [[PJThread alloc] initWithBlock:^{
        NSTimer *timer7 = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod7) userInfo:nil repeats:YES];
        //[NSRunLoop currentRunLoop]会获取当前的RunLoop,**如果当前没有则会创建一个**
        NSRunLoop *runLoop2 = [NSRunLoop currentRunLoop];
        //在NSRunLoopCommonModes下也是可以的,但是苹果官方推荐NSDefaultRunLoopMode,在runLoop1下已经不会受到UI滚动的影响了
        [runLoop2 addTimer:timer7 forMode:NSDefaultRunLoopMode];
        //runUntilDate表示每隔多久执行一次runLoop,我们设置点击屏幕就让finished为真YES,看能不能达到退出循环打印"runLoop2执行之后!"(RunLoop执行速度很快,故设置0.01够了),实际运行时可以打印"runLoop2执行之后!",并且也打印"PJThread : pjThread3 被销毁了!"
        while(!weakSelf.finished){
            [runLoop2 runUntilDate:[NSDate dateWithTimeIntervalSinceReferenceDate:0.01]];
        }
        NSLog(@"runLoop2执行之后!");
    }];
    [pjThread3 setName:@"pjThread3"];
    //线程跑起来
    [pjThread3 start];
    
    //到此暂时结束,线程和RunLoop的关系是非常密切的,一个线程要保持一致运行部退出需要RunLoop来帮助实现
}

- (void)timeMethod1{
    NSLog(@"timeMethod1");
}

- (void)timeMethod2{
    NSLog(@"timeMethod2");
}

- (void)timeMethod3{
    NSLog(@"timeMethod3");
}

- (void)timeMethod4{
    [NSThread sleepForTimeInterval:1.0];
    NSLog(@"timeMethod4");
}

- (void)timeMethod5{
    [NSThread sleepForTimeInterval:1.0];
    NSLog(@"timeMethod5");
}

- (void)timeMethod6{
    [NSThread sleepForTimeInterval:1.0];
    NSLog(@"timeMethod6");
}

- (void)timeMethod7{
    [NSThread sleepForTimeInterval:1.0];
    NSLog(@"timeMethod7");
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    self.finished = YES;
}

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,386评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,016评论 4 62
  • 如果没有微信公众号 我会在哪里 也许 就一直懒惰的安逸中死去 没有激情 没有创造的灵魂 行尸走肉 但是 我是何等的...
    刘娇阅读 365评论 0 4
  • 当我读这本书时,就有了思考,“活着”到底是为了什么?有人说,活着是为了更好的享受生活;有人说,活着是为了...
    苑殇阅读 377评论 2 0